ClassLoader和Class.forName区别
先了解一下类加载的过程
加载阶段:
1.通过类的权限定名加载该类的二进制字节流。
2.将这个类的二进制字节流转换为方法区的运行是数据结构。
3.在方法区中生成一个代表这个类的java.lang.Class对象。
验证阶段:
验证阶段主要是对虚拟机加载的二进制文件规则进行校验,判断当前信息是否符合虚拟机要求。
准备阶段:
主要是为类变量分配内存,并未类变量复制初始值(这里指的初始值为该类型的0值)。比如public static int a = 123;那么此时为a变量分配内存,并且复制为0;如果为public static final int a = 123;则会在编译的时候放在常量池中。
解析阶段:
解析阶段是将常量池内的符号引用替换为直接引用。符号引用是指一组描述符所引用的目标,直接引用是指指向目标的指针偏移量或句柄等。
初始化阶段:
编译器会将静态代码块代码和静态变量合成到<clinit>方法中。在初始化阶段就会被调用。<clinit>也就是初始化类的方法。
Class.forName 加载类
Class.forName(className);根据类的权限定名加载类,如果不指定参数,就会调用下面的方法,将会执行类的初始化。
@CallerSensitive
public static Class<?> forName(String className)
throws ClassNotFoundException {
Class<?> caller = Reflection.getCallerClass();
return forName0(className, true, ClassLoader.getClassLoader(caller), caller);
}
Class.forName(name, initialize, loader);可以指定是否执行类初始化和类的加载器。 下面为案例:如果不进行初始化就不会执行代码块的内容。
package sunhongliang;
public class ClassLoaderTest {
public static void main(String[] args) throws Exception {
Class.forName("sunhongliang.Student");
Class.forName("sunhongliang.School", false,
ClassLoaderTest.class.getClassLoader());
}
}
class Student {
static {
System.out.println("init student ...");
}
public void method1() {
System.out.println("method1 ...");
}
}
class School {
static {
System.out.println("init school ...");
}
public void method2() {
System.out.println("method2 ...");
}
}
输出结果:
init student ...
ClassLoader 加载类
ClassLoader.loadClass(name),加载类的时候会调用下面的方法。下面false表示不进行连接,也就是不会初始化操作。
public Class<?> loadClass(String name) throws ClassNotFoundException {
return loadClass(name, false);
}
package sunhongliang;
public class ClassLoaderTest {
public static void main(String[] args) throws Exception {
ClassLoader systemClassLoader = ClassLoader.getSystemClassLoader();
systemClassLoader.loadClass("sunhongliang.Student");
}
}
class Student {
static {
System.out.println("init student ...");
}
public void method1() {
System.out.println("method1 ...");
}
}
输出结果为空