背景知识:JVM怎么运行java代码?
Java源文件->编译器->字节码文件(.class文件)->JVM->机器码
1.类加载过程:
Jdk通过指令,将.java文件编译成.class文件,.class文件通过自定义类加载器(ClassLoader)使用“双亲委派机制”加载.class文件。
加载->链接(验证,准备,解析)->初始化->调用
- 加载:在内存中生成java.lang.Class对象。
- 验证:确保Class文件的字节流中包含的信息是否符合当前虚拟机的要求。
- 准备:即在方法区中分配这些变量所使 用的内存空间。
比如一个类变量定义为public static int v=8080
,实际上在准备阶段,v初始化=0,8080真正被赋值给v,是在初始化阶段时调用类构造器的方法
注意:如果定义为public static final int v=8080
,则v是一个常量,在准备阶段就完成初始化 - 解析:指虚拟机将常量池中的符号引用替换为直接引用的过程。
- 初始化:是执行类构造器方法的过程
2.引起类加载器的行为:
首先确保类被加载和链接
-创建类的实例,new 一个对象,getstatic,putstatic,invokestatic
-访问类或接口的静态变量,
-调用类的静态方法,Math.random()
-反射
-初始化一个类的子类
-JVM启动时标明的启动类
3.切入正题,研究通过反射来实现类加载的两种方式与区别
-两者都可以用来对类进行加载
-class.forName(“className”)除了将类的.class文件加载到jvm中,还会执行static静态块
-ClassLoader.loadClass(“className”),将.class文件加载到JVM中,只有在newInstance才会执行static块
Class.forName方法源码,测试代码,与结果分析
loadClass方法源码,测试代码,与结果分析
测试结果很清楚的展示了两种方式加载类的区别,主要是因为在class.forName()中进行了类的初始化,这里要分清
类的初始化与类的实例化
那么现在我们来测试类的实例化,即加上line.newInstance();
这里只对loadClass进行测试,forName()也是会执行同样的实例化结果