Java哪些情况下必须对类进行“初始化”
- 遇到new、getstatic、putstatic或invokestatic这四条字节码指令时,如果类没有进行初始化,则需要先触发其初始化。生成这四条指令的常见场景是:
- 使用new关键字实例化对象的时候
- 读取或设置一个类的静态字段的时候(被final修饰、已在编译期把结果放入常量池的静态字段除外)
- 调用一个类的静态方法的时候
- 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行初始化,则需要先触发其初始化
- 当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其初始化
- 当虚拟机启动时,用户需要指定一个要执行的主类(包含main方法的那个类),虚拟机会先初始化这个主类
- 当使用JDK1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行初始化,则需要先触发其初始化
注意:有且只有上面这五种情况下才会触发类的初始化
这些情况下不会对类进行初始化:
- 通过子类引用父类的静态字段,不会导致子类的初始化
- 通过数组定义来引用类,不会触发此类的初始化
- 常量在编译阶段会存入调用类的常量池中,本质上没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化
实例如下:
public class SuperClass {
static {
System.out.println("SuperClass init!!");
}
public static int value = 123;
}
public class SubClass extends SuperClass {
static {
System.out.println("SubClass init!!");
}
}
public class ConstClass {
static {
System.out.println("ConstClass init!!");
}
public static final String HELLOWORLD = "hello world!";
}
// 测试
public class NotInitialization {
public static void main(String args[]) {
/**
* 通过子类引用父类的静态字段,不会导致子类的初始化
* 打印结果:
* SuperClass init!!
* 123
*/
// System.out.println(SubClass.value);
/**
* 通过数组定义来引用类,不会触发此类的初始化
* 打印结果:
* 无
*/
// SuperClass[] sca = new SuperClass[10];
/**
* 常量在编译阶段会存入调用类的常量池中,本质上没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化
* 打印结果:
* hello world!
*/
System.out.println(ConstClass.HELLOWORLD);
}
}