在讲这两个方法之前,我先说一下,类的加载方式。
一个类被加载出来,是有两种方式的,分为隐式加载和显式加载:
-
隐式加载:new
-
显式加载:loadClass,forName等
隐式加载就不多说明了,通过new一个class来获得到类的实例,并且相对于显式加载更优秀的地方是,可以直接调用有参构造;
但是显式调用的原理则不是这么简单的,当我们获取到class对象之后,需要调用class对象的new Instance()方法,才能获得到类的实例,而且我们调用new Instance()的时候,无法进行参数的填充。
介绍完隐式加载和显式加载之后,我们就来主要的来说一下loadClass和forName这两个显式加载,究竟有什么区别。
首先,当我们调用这两个方法的时候,我们都可以知道类对应的全部属性和方法,对于一个任意的对象,都可以调用其属性和方法。
我们先从类的装载过程来入手解决这个问题:
那么,什么是装载呢?
也就是一个Class对象的生成过程。其中,Class对象的装载,一共分为三步:
- 加载
通过ClassLoader的loadClass方法,将类的字节码加载到内存中,将这些类型数据,转换成运行时方法区的类型数据,在运行的时候,就会自然生成一个.class,成为访问当前这个java文件的入口。
- 链接
检验.class文件是否可以通过编译,并且同时检查他的安全性;再去准备为类变量分配存储空间并设置类变量的初始值;最后解析JVM将常量池内的符号引用转换为直接引用。
- 初始化
执行类变量相关的赋值和静态的代码块。
从本质上来讲,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的结果就是只完成第一步,检查编译是否可以通过,所以结果很简单, 确实无法输出任何东西。
然后再执行这段代码,按照上面的说明,初始化的步骤应该完成,所以静态块也应该加载出来。
然后我们来验证一下我们的想法:
的确是没有问题的。