剑指Offer(JVM)——loadClass和forName的区别

在讲这两个方法之前,我先说一下,类的加载方式。

一个类被加载出来,是有两种方式的,分为隐式加载和显式加载

  • 隐式加载:new

  • 显式加载:loadClass,forName等

隐式加载就不多说明了,通过new一个class来获得到类的实例,并且相对于显式加载更优秀的地方是,可以直接调用有参构造;

但是显式调用的原理则不是这么简单的,当我们获取到class对象之后,需要调用class对象的new Instance()方法,才能获得到类的实例,而且我们调用new Instance()的时候,无法进行参数的填充。

介绍完隐式加载和显式加载之后,我们就来主要的来说一下loadClass和forName这两个显式加载,究竟有什么区别

首先,当我们调用这两个方法的时候,我们都可以知道类对应的全部属性和方法,对于一个任意的对象,都可以调用其属性和方法。

我们先从类的装载过程来入手解决这个问题:

那么,什么是装载呢?

也就是一个Class对象的生成过程。其中,Class对象的装载,一共分为三步:

  1. 加载

通过ClassLoader的loadClass方法,将类的字节码加载到内存中,将这些类型数据,转换成运行时方法区的类型数据,在运行的时候,就会自然生成一个.class,成为访问当前这个java文件的入口。

  1. 链接

检验.class文件是否可以通过编译,并且同时检查他的安全性;再去准备为类变量分配存储空间并设置类变量的初始值;最后解析JVM将常量池内的符号引用转换为直接引用。

  1. 初始化

执行类变量相关的赋值和静态的代码块。

从本质上来讲,Class.forName得到的class是已经初始化完成的,而ClassLoader.loadClass得到的class是还没有完成链接的。

为了更加凸显它们的区别,我们来用代码来体现一下:

首先我们编写一个带静态块的Robot类:

public class Robot {

    private String name;
    public void sayHi(String helloSentence) {
        System.out.println(helloSentence + "" + name);
    }

    private String throwHello(String tag) {
        return "hello" + "" + tag;
    }

	static{
		System.out.println("hello Robot");
	}

}

然后我们执行这段代码,按照我们上面的解释,使用loadClass的结果就是只完成第一步,检查编译是否可以通过,所以结果很简单, 确实无法输出任何东西。
在这里插入图片描述
然后再执行这段代码,按照上面的说明,初始化的步骤应该完成,所以静态块也应该加载出来。
在这里插入图片描述

然后我们来验证一下我们的想法:
在这里插入图片描述
在这里插入图片描述

的确是没有问题的。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值