本文主要介绍类初始化的场景和被动引用案例;
一、简介
- 本文主要介绍类初始化的场景和被动引用案例;
二、类初始化场景
- 遇到new、getstatic、putstatic或invokestatic这四条字节码指令时,如果类型没有进行过初始化,则需要先触发其初始化阶段。能够生成这四条指令的典型java代码场景有:a)使用new关键字实例化对象的时候;b)读取或设置一个类型的静态字段(被final修饰、已在编译期把结果放入常量池的静态字段除外)的时候;c)调用一个类型的静态方法的时候;
- 使用Java.lang.reflact包的方法对类型进行反射调用的时候,如果类型没有进行过初始化,则需要先触发其初始化;
- 当初始化类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化;
- 当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类;
- 当使用JDK7新加入的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果为REF_getStatic、REF_putStatic、REF_invokeStatic、REF_newInvokeSpecial四种类型的方法句柄,并且这个方法句柄对应的类没有进行过初始化,则需要先触发其初始化;
- 当一个接口中定义了JDK8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化;
- 注:《Java虚拟机规范》有且只有上述六种才会触发类型的初始化;
三、接口初始化场景
- 接口也有初始化过程,这点与类是一致的,类的静态语句块是用“static{}”来输出初始化信息的,而接口中不能使用“static{}”语句块,但编译器仍然会为接口生成""类构造器,用于初始化接口中所定义的成员变量;
- 接口与类真正有所区别的是上面讲述的六种有且仅有需要触发初始化场景的第三种:当一个类在初始化时,要求其父类全部都已经初始化过了,但是一个接口在初始化时,并不要求其父接口全部都完成了初始化,只有在真正使用到父接口的时候(如引用接口中定义的常量)才会初始化;
四、被动引用案例
- 上述六种场景的行为称为对一个类型进行主动引用,除此之外,所有引用类型的方式都称为被动引用,不会触发类型的初始化;
4.1 通过子类引用父类的静态字段,不会导致子类初始化
4.2 通过数组定义来引用类,不会触发此类的初始化
4.3 引用常量不会触发定义常量的类的初始化