[b]一、定义:虚拟机把描述类的数据从class文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制。
与需要在编译时进行连接工作的语言不同,在Java语言里面,类型的加载、连接和初始化过程都是在程序运行期间完成的,这种策略虽然会令类加载时稍微增加一些性能开销,但是会为Java应用程序提供高度的灵活性,Java里天生可以动态扩展的语言特性就是依赖运行期动态加载和动态连接这个特点实现的。
[/b]
[b]二、类加载的时机[/b]
类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载、验证、准备、解析、初始化、使用和卸载7个阶段。其中验证、准备、解析3个部分称为连接(Linking)。
1)遇到new、getstatic、putstatic、invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。常见的Java代码场景为:new、读取或设置一个类的静态字段(被final修饰,已在编译期把结果放入常量池的静态字段除外),以及调用一个类的静态方法的时候。
2)使用java.lang.relect包中的方法对类进行反射调用。
3)初始化一个类的时候,如果其父类还没有进行初始化,则需要先触发其父类的初始化。
4)虚拟机启动,需要先指定一个要执行的主类(包含main方法的那个类),虚拟机会先初始化这个主类。
5)使用JDK1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。
[b]注意:
1、通过数组定义来引用类、不会触发此类的初始化。
2、常量在编译阶段会存入调用类的常量池中,本质上并没有直接引用到定义常量的类,因此不会触发定义常量类的初始化。[/b]
[b]三、类加载的过程[/b]
加载(将二进制加载到内存,访问入口设置)->验证(防篡改)->准备(在方法区内为类变量分配内存并设置初始值)->解析(将符号引用替换为直接引用)->初始化(执行构造器,初始化对象)
与需要在编译时进行连接工作的语言不同,在Java语言里面,类型的加载、连接和初始化过程都是在程序运行期间完成的,这种策略虽然会令类加载时稍微增加一些性能开销,但是会为Java应用程序提供高度的灵活性,Java里天生可以动态扩展的语言特性就是依赖运行期动态加载和动态连接这个特点实现的。
[/b]
[b]二、类加载的时机[/b]
类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载、验证、准备、解析、初始化、使用和卸载7个阶段。其中验证、准备、解析3个部分称为连接(Linking)。
1)遇到new、getstatic、putstatic、invokestatic这4条字节码指令时,如果类没有进行过初始化,则需要先触发其初始化。常见的Java代码场景为:new、读取或设置一个类的静态字段(被final修饰,已在编译期把结果放入常量池的静态字段除外),以及调用一个类的静态方法的时候。
2)使用java.lang.relect包中的方法对类进行反射调用。
3)初始化一个类的时候,如果其父类还没有进行初始化,则需要先触发其父类的初始化。
4)虚拟机启动,需要先指定一个要执行的主类(包含main方法的那个类),虚拟机会先初始化这个主类。
5)使用JDK1.7的动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getStatic、REF_putStatic、REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行过初始化,则需要先触发其初始化。
[b]注意:
1、通过数组定义来引用类、不会触发此类的初始化。
2、常量在编译阶段会存入调用类的常量池中,本质上并没有直接引用到定义常量的类,因此不会触发定义常量类的初始化。[/b]
[b]三、类加载的过程[/b]
加载(将二进制加载到内存,访问入口设置)->验证(防篡改)->准备(在方法区内为类变量分配内存并设置初始值)->解析(将符号引用替换为直接引用)->初始化(执行构造器,初始化对象)