
原文链接:JVM系列:(六)JVM类加载步骤
上一章我们理解了class文件存储结构,在class文件中形容的各种信息,最终都需要加载到虚拟机中之后才能被运行和使用。而虚拟机是如何加载这些class文件的?
虚拟机把形容类的数据从class文件加载到内存,并对数据进行校验、转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制。
JVM中类型的加载和连接过程都是在程序运行期间完成的,这样会在类加载时略微添加少量性能开销,但是却能为Java应用程序提供高度的灵活性,Java中天生可以动态扩展的语言特性就是依赖运行期动态加载和动态连接这个特性实现的。例如,假如编写一个使用接口的应用程序,可以等到运行时再指定其实际的实现。
类的完整生命周期包括:加载(loading)、验证(verification)、准备(preparation)、解析(resolution)、初始化(initialization)、使用(using)、卸载(unloading)。其中验证、准备、解析这三个步骤可以统称为连接(linking)。如下图所示:
类的生命周期今天我们要说的类加载步骤不包括最后两步的使用和卸载,主要讲解步骤为:加载、验证、准备、解析和初始化。
加载阶段是虚拟机类加载的第一步骤,在加载阶段,虚拟机需要完成以下三件事情:
虚拟机规范的这三点要求实际上并不具体,例如“通过一个类的全限定名来获取定义此类的二进制字节流”,并没有指明二进制字节流要从一个 class 文件中获取,没有指明要从哪里获取及怎么获取,例如:
相对于类加载过程的其余阶段,加载阶段是开发期可控性最强的阶段,加载阶段既可以使用系统提供的类加载器来完成,也可以由客户自己设置的类加载器去完成,开发人员可以定义自己的类加载器去控制字节流的获取方式。
加载阶段完成后,虚拟机外部的二进制字节流就按照虚拟机所需的格式存储在方法区中,而后在Java堆内存中实例化一个java.lang.Class类的对象,这个对象将作为程序访问方法区中的这些类型数据的外部接口。
这一阶段的目的是为了确保class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。我们前面已经说过,class文件不肯定要求用Java源码编译而来,可以使用任何途径,包括用十六进制编辑器直接编写产生class文件。虚拟机假如不检查输入的字节流,对其完全信任的话,很可能会由于载入了有害的字节流而导致系统崩溃,所以验证是虚拟机对自身保护的一项重要工作。
大致上验证会完成下面四个阶段的检验过程:
准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些内存都将在方法区中进行分配。
这个阶段中需要注意的两点:
解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。包括:
类初始化阶段是类加载过程的最后一步,前面的类加载过程中,除了在加载阶段客户应用程序可以通过自己设置类加载器参加之外,其他动作完全由虚拟机主导和控制。到了初始化阶段,才真正开始执行类中定义的Java程序代码(编译后的字节码)。
在准备阶段,变量已经被赋过一次变量默认的初始值,而在初始化阶段,则会根据程序中编写的具体值去初始化类变量和其余资源。
对于初始化阶段,虚拟机规范严格规定了有且只有四种情况必需立即对类进行“初始化”(加载、验证、准备自然需要在此之前开始);
有且仅有以上四种情况会执行类的初始化。
类的加载步骤总结来说就是,先把远程或者磁盘的class文件加载到内存中,并在Java堆内存生成该类的class对象以供外部访问、再对加载到的数据进行验证,校验能否符合JVM定义的class文件结构规范、其次再对定义的类变量赋初始值、最后就是对类变量进行赋定义值以及相关初始化工作。
接下来我们将详情:
扫码关注有惊喜