Classloader中loadClass()方法和Class.forName()区别

时间:2021-06-10 19:29:37

Classloader中loadClass()方法和Class.forName()都能得到一个class对象,那这两者得到的class对象有什么区别呢

1、java类装载的过程

Java类装载有三个步骤:

装载:通过累的全限定名获取二进制字节流,将二进制字节流转换成方法区中的运行时数据结构,在内存中生成Java.lang.class对象; 

链接:执行下面的校验、准备和解析步骤,其中解析步骤是可以选择的; 

  校验:检查导入类或接口的二进制数据的正确性;(文件格式验证,元数据验证,字节码验证,符号引用验证) 

  准备:给类的静态变量分配并初始化存储空间;

  解析:将常量池中的符号引用转成直接引用;

初始化:激活类的静态变量的初始化Java代码和静态Java代码块,并初始化程序员设置的变量值。
接下来来讨论这两者的区别

2、Classloader中loadClass()方法和Class.forName()区别

先来看段程序

Person类:

package com.pjf;

public class Person {
static {
System.out.println("Initializing");
}
}

测试类:

Class cls1 = Class.forName("com.pjf.Person");
System.out.println("!");
Person person1 = (Person) cls1.newInstance();
System.out.println(person1.getClass().getName()); //输出结果
Initializing
!
com.pjf.Person

测试类2

Dog dog =new Dog();
ClassLoader loader = dog.getClass().getClassLoader() ;
Class cls = loader.loadClass("com.pjf.Person");
System.out.println("!");
Person person1 = (Person) cls.newInstance();
System.out.println(person1.getClass().getName());
//输出结果
!
Initializing
com.pjf.Person

结论:可以看到输出结果Initializing和!输出顺序反了,这是因为Class.forName得到的class是已经初始化完成的,Classloder.loaderClass得到的class是还没有链接的

源码分析:

Class.forName(className)方法,内部实际调用的方法是  Class.forName(className,true,classloader);

第2个boolean参数表示类是否需要初始化,  Class.forName(className)默认是需要初始化。

一旦初始化,就会触发目标对象的 static块代码执行,static参数也也会被再次初始化。

ClassLoader.loadClass(className)方法,内部实际调用的方法是  ClassLoader.loadClass(className,false);

第2个 boolean参数,表示目标对象是否进行链接,false表示不进行链接,由上面介绍可以,

不进行链接意味着不进行包括初始化等一些列步骤,那么静态块和静态对象就不会得到执行