接口与类真正有所区别的是前面讲述的六种“有且仅有”需要开始初始化场景中的第三种:当一个类在初始化时,要求其父类全部都已经初始化过了,但是一个接口在初始化时,并不要求其父接口全部都完成了初始化,只有在真正使用到父接口的时候(如引用接口中定义的常量)才会初始化。 ——《深入理解Java虚拟机(第三版):JVM高级特性与最佳实践》
但是文中又指出:
引用类中的常量不会导致类被初始化,因为编译阶段已经将常量移动到常量池中了
有些疑惑,是不是冲突呢?不过仔细看了下之前的内容,文中也提到
常量池中只能引用到基本类型和String类型的字面量
通过这句话说明:
类和接口在被引用常量的时候是否被初始化,取决于这个常量能够在编译时被放进常量池中(排除不支持的类型和运行时常量)。
最后
类初始化(主动引用)几类情况:
- 遇到new、getstatic、putstatic或invokestatic这四条字节码指令时,如果类型没有进行过初始化,则需要先触发其初始化阶段:
- 使用new关键字实例化对象时
- 读取或设置一个类型的静态字段的时候,(被final修饰、已在编译期把结果放入常量池的静态字段除外)
- 调用一个类型的静态方法时
- 使用java.lang.reflect包的方法对类型进行反射调用的时候,如果类型没有初始化,需要先初始化
- 当初始化类时发现其父类还没有初始化,则需要先触发其父类进行初始化
- 当虚拟机启动时,用户需要指定一个要执行的主类(包含main方法的类),虚拟机会先初始化主类。
- 当时用JDK7新加入的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果为REFgetStatic、REFputStatic、REFinvokeStatic、REFnewInvokeSpecial四种类型的方法句柄,并且这个句柄对应的类还没有进行过初始化,则需要进行初始化。
- 当一个接口中定义了JDK8新加入的默认方法时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化。