1.概述
虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验、转换分析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制。
因为Java语言是可以动态扩展的语言,类型的加载、连接和初始化过程中都是在程序运行期间完成的,而其动态扩展的特性是依赖运行期动态加载和动态连接的特点实现的,这就需要类加载时稍微增加一些性能开销。
2.类加载的时机
类从被加载到虚拟机内存开始,到卸载出内存为止,他的生命周期包括:加载、验证、准备、解析、初始化、使用、卸载7个阶段。其中验证、准备、解析3个部分统称为连接。
其中加载、验证、准备、初始化、卸载这5个阶段是按照顺序按部就班的开始的,但不代表按部就班的进行与结束,而各个阶段互相交叉地混合式进行,通常会在一个阶段执行时调用或者激活另一个阶段。其中解析阶段比较特殊,它可能在初始化阶段后再开始,为了支持Java语言运行时绑定(动态绑定)。
进行初始化的6种情况:
①遇到new、getstatic、putstatic、invokestatic这4条字节码指令时,若类没有进行初始化,会立即触发初始化。而用到这写指令的情况通常为:使用new关键字实例化对象时、读取或者设置一个类型的静态字段(被final修饰、已在编译器把结果放入常量池的静态字段除外)以及调用一个类型的静态方法的时候。
②使用java.lang.reflect包的方法对类型进行反射调用的时候,如果类型没有进行过初始化,则需要先触发其初始化。
③当初始化类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。(若通过子类引用父类中定义的静态字段,只会触发父类的初始化而不会触发子类的初始化。)
④当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
⑤当使用JDK 7新加入的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果为REF_getStatic、REF_putStatic、REF_invokeStatic、REF_newInvokeSpecial四种类型的方法句 柄,并且这个方法句柄对应的类没有进行过初始化,则需要先触发其初始化。
⑥当一个接口中定义了JDK 8新加入的默认方法(被default关键字修饰的接口方法)时,如果有这个接口的实现类发生了初始化,那该接口要在其之前被初始化。
这6种情况的行为称为对一个类进行主动引用,除此之外所有引用类的方式都不会触发初始化,称为被动引用。