1、概述
类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载(Loading)、验证(Verification)、准备(Preparation)、解析(Resolution)、初始化(Initialization)、使用(Useing)、卸载(Unloading)7个阶段。其中验证、准备和解析3个部分统称为连接(Linking),这7个阶段的发生顺序如图所示:

加载、验证、准备、初始化和卸载这5个阶段的顺序是确定的,类的加载过程,必须按照这种顺序按部就班地开始,而解析阶段则不一定:它在某些情况下可以在初始化阶段之后再开始,这是为了支持Java语言的运行时绑定(也称为动态绑定或晚期绑定) 。
对于初始化阶段,虚拟机规范则是严格规定了有且只有5种情况必须立即对类进行“初始化(执行<clinit>方法阶段)”( 而加载、 验证、 准备自然需要在此之前开始):
- 遇到
new、getstatic、putstatic或invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。 - 使用
java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。 - 当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
- 当虚拟机启动时,用户需要指定一个要执行的主类( 包含main()方法的那个类),虚拟机会先初始化这个主类。
- 当使用JDK 1.7的动态语言支持时,如果一个
java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄, 并且这个方法句柄所对应的类没有进行过初始化, 则需要先触发其初始化。
这五种情况称为对一个类进行主动引用。其余情况都不会触发初始化,称为被动引用。
2、主动引用
首先准备两个类用户测试其是否初始化。
SuperClass
/**
* @author lastwhisper
*/
public class SuperClass {
public static int value = 123;
static {
System.out.println("SuperClass static code init!");
}
public SuperClass() {
System.out.println("SuperClass constructor init! ");
}
public static int getValue() {
return value;
}
public static void setValue(int value) {
SuperClass.value = value;
}
}
SubClass 继承自SuperClass
/**
* @author lastwhisper
*/
public class SubClass extends SuperClass {
public static int subvalue = 456;
static {
System.out.println("SubClass static code init!");
}
public SubClass() {
System.out.println("SubClass constructor init! ");
}
public static int getSubvalue() {
return subvalue

本文详细阐述了Java类加载的七个阶段,重点介绍了主动引用(如new、getstatic等)和被动引用如何触发或不触发类的初始化,通过示例展示了这两种引用方式在不同场景下的行为。
最低0.47元/天 解锁文章
1746

被折叠的 条评论
为什么被折叠?



