jvm规范严格规定了有且仅有六种情况必须立即对类进行初始化:
- 遇到new、getstatic、putstatic、invokestatic这四条字节码指令时,如果类型没有进行过初始化,则需要先触发其初始化阶段。能够生成四条指令的典型java代码场景有:
a. 使用new关键字实例化对象的时候
b. 读取或设置一个类型的静态字段(final修饰已在编译期把结果放入常量池的静态字段除外)的时候
c. 调用一个类型的静态方法 - 使用java.lang.reflect包的方法对类型进行反射调用的时候,如果类型没有进行过初始化,则需要先触发其初始化
- 当初始化类的时候,如果发现其父类还没有进行过初始化,则需要先触发父类的初始化
- 当虚拟机启动时,用户需要指定一个要执行的主类(包括main方法的那个类),虚拟机会先初始化这个主类
- 当jdk7新加入的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果为下面四种类型的方法句柄,并且这个方法句柄对应的类没有进行过初始化
a. REF_getStatic
b. REF_putStatic
c. REF_invokeStatic
d. REF_newInvokeSpecial - 当一个接口中定义了JDK8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化。
这六种场景中的行为称为对一个类型进行主动引用。
除此之外,所用引用类型的方式都不会触发初始化,称为被动引用。
接口初始化与类初始化有不同:
接口中没有static{},但是初始化时会有<clinit>类构造方法,用于初始化接口中定义的成员变量。
类初始化条件第三条,类初始化时必须初始化所有父类,但是在接口实例化时,并不需要实例化所有父接口,只有真正使用父接口的时候(如引用接口中定义的变量)才会初始化。
网友评论