深入理解JAVA虚拟机系列之类加载的过程
一、简介
- 本文主要介绍类加载的过程的几个阶段;
二、类加载过程

- 一个类型从被加载到虚拟机内存开始,到卸载出内存为止,它的整个生命周期将会经历以上几个阶段;
- 加载、验证、准备、初始化和卸载这五个阶段的顺序是固定的,而解析阶段则不一定:它在某些情况下可以在初始化阶段之后再开始,这是为了支持java语言的运行时绑定特性(也称为动态绑定或晚期绑定);
- 请注意:这里写的是按部就班的开始,而不是按部就班的进行或按部就班的完成,强调这点是因为这些阶段通常都是互相交叉地混合进行的,会在一个阶段执行的过程中调用、激活另一个阶段;
2.1 加载
2.1.1 原理
- 通过一个类的全限定名来获取定义此类的二进制字节流;
- 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构;
- 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据访问入口;
2.2 验证
2.2.1 文件格式验证
2.2.1.1 原理
- 主要是验证字节流是否符合Class文件格式的规范,并且能被当前版本的虚拟机处理;
- 该阶段的验证是基于二进制字节流进行的,只有通过了这个阶段的验证后,这段字节流才被允许进入java虚拟机内存的方法区中进行存储,后面三个验证阶段都是基于方法区的存储结构上进行的,不再直接读取、操作字节流了;
2.2.1.2 具体验证点
- 是否以魔数0xCAFEBABE开头;
- 主、次版本号是否在当前Java虚拟机接受范围之内;
- 常量池的常量中是否有不被支持的常量类型(检查常量tag标志);
- 指向常量的各种索引值中是否有指向不存在的常量或不符合类型的常量;
- CONSTANT_Utf8_info型的常量中是否有部分胡UTF-8编码的数据;
- Class文件中各个部分及文件本身是否有被删除的或附加的其他信息;
- 。。。
2.2.2 元数据验证
2.2.2.1 原理
- 该阶段是对字节码描述的信息进行语义分析,主要目的是对类的元数据信息进行语义校验,以保证其描述的信息符合《java语言规范》的要求;
2.2.2.2 具体验证点
- 该类是否有父类(除了java.lang.Object,所有的类都应当有父类);
- 该类的父类是否继承了不允许被继承的类(被final修饰的类);
- 如果这个类不是抽象类,是否实现了其父类或接口之中要求实现的所有方法;
- 类中的字段、方法是否与父类产生矛盾(例如覆盖了父类的final字段,或者出现不符合规则的方法重载,例如方法参数都一直,但返回值类型却不同等);
- 。。。
2.2.3 字节码验证
2.2.3.1 原理
- 主要目的是通过数据流分析和控制流分析,确定程序语义是合法、符合逻辑的;
- 在上阶段对元数据信息中的数据类型校验完毕之后,该阶段就要对类的方法体(Class文件中的Code属性)进行校验分析,保证被校验类的方法在运行时不会做出危害虚拟机安全的行为;
- 如果一个类型中有方法体的字节码没有通过字节码验证,那它肯定是有问题的。但如果一个方法体通过了字节码验证,也仍然不能保证它一定就是安全的;
2.2.3.2 具体验证点
- 保证任意时刻操作数栈的数据类型与指令代码序列都能配合工作,例如不会出现类似于“在操作栈放置了一个int类型的数据,使用时却按long类型来加载入本地变量表中”这样的情况;
- 保证任何跳转指令都不会跳转到方法体以外的字节码指令上;
- 保证方法体中的类型转换总是有效的,例如可以把一个子类对象赋值给父类数据类型,这是安全的,但是把父类对象赋值给子类数据类型,甚至把对象赋值给与它毫无继承关系、完全不相关的一个数据类型,则是危险和不合法的;
- 。。。
2.2.4 符号引用验证
2.2.4.1 原理
- 该阶段的校验行为发生在虚拟机将符号引用转换为直接引用的时候,这个转化动作将在连接的第三阶段---解析阶段中发生;
- 符号引用验证可以看作是对类自身以外(常量池中的个汇总符号引用)的各类信息进行匹配性校验,通俗来说就是,该类是否缺少或者被禁止访问它依赖的某些外部类、方法、字段等资源;
- 符号引用验证的主要目的是确保解析行为能正常执行,如果无法通过符号引用验证,java虚拟机将会抛出一个java.lang.IncompatibleClassChangeError的子类异常,典型的如:java.lang.IllegalAccessError、java.lang.NoSuchFieldError、java.langlNoSuchMethodError等;
2.2.4.2 具体验证点
- 符号引用中通过字符串描述的全限定名是否能找到对应的类;
- 在指定类中是否存在符合方法的字段描述符及简单名称所描述的方法和字段;
- 符号引用中的类、字段、方法的可访问性(private、protected、public、
)是否可被当前类访问; - 。。。
2.2.5 关闭验证
- 验证阶段对于虚拟机的类加载机制来说是一个非常重要的、但却不是必须要执行的阶段,因为验证阶段只有通过或者不通过的差别,只要通过了验证,其后就对程序运行期没有任何影响了。如果程序运行的全部代码(包括自己编写的、第三方包中的、从外部加载的、动态生成的等所有代码)都已经被反复使用和验证过,在生产环境的实施阶段就可以考虑使用-Xverify:none参数来关闭大部分的类验证措施,以缩短虚拟机类加载的时间;
2.3 准备
2.3.1 原理
- 准备阶段是正式为类中定义的变量(即静态变量,被static修饰的变量)分配内存并设置类变量的初始值的阶段;
- 该阶段主要针对类变量,而非实例变量。是里边俩你个将会在对象实例化的时候随着对象一起分配在java堆中;
- 注意分配的类变量的初始值一般是变量所对应类型的零值,如int类型的为0;
- 如果类字段的字段属性表中存在ConstantValue属性,即是被final和static修饰的常量,则会在准备阶段直接赋值为设置的初始值;
2.4 解析
2.4.1 原理
-
符号引用:符号引用以一组符号来描述所引用的目标,符号可以是任何形式的字面量,只要使用时能无歧义地定位到目标即可。符号引用与虚拟机实现的内存布局无关,引用的目标并不一定是已经加载到内存当中的内容。各种虚拟机实现的内存布局可以各不相同,但是它们能接受的符号引用必须都是一致的,因为符号引用的字面量形式明确定义在《java虚拟机规范》的Class文件格式中,如CONSTANT_Class_info,CONSTANT_Fieldref_info等;
-
直接引用:直接引用是可以直接指向目标的指针、相对偏移量或者是一个能间接定位到目标的句柄。直接引用是和虚拟机实现的内存布局直接相关的,同一个符号引用在不同虚拟机实例上翻译出来的直接引用一般不会相同。如果有了直接引用,那引用的目标必定已经在虚拟机的内存中存在;
-
解析阶段是java虚拟机将常量池内的符号引用转换为直接引用的过程;
-
《Java虚拟机规范》之中并未规定解析阶段发生的具体时间,只要求了在执行anewarray、checkcast、getfield、getstatic、instanceof、invokedynamic、invokeinterface、invokespecial、invokestatic、invokevirtual、ldc、ldc_w、ldc2_w、multianewarray、new、putfield和putstatic这17个用于操作符号引用的字节码指令之前,先对它们所使用的符号引用进行解析。所以虚拟机实现可以根据需要来自行判断,到底是在类被类加载时就对常量池中的符号引用进行解析,还是等到一个符号引用将要被使用前才去解析它;
-
类似地,对方法或者字段的访问,也会在解析阶段中对它们的可访问性(public、protected、private、
)进行检查,至于其中的约束规则已经是java语言的基本常识了; -
对同一个符号引用进行多次解析请求是很常见的事情,除invokedynamic指令以外,虚拟机实现可以对第一次解析的结果进行缓存,譬如在运行时直接引用常量池中的记录,并把常量标识为已解析状态,从而避免解析动作重复执行。无论是否执政执行了多次解析动作,java虚拟机都需要保证的是在同一个实体中,如果一个符号引用之前已经被成功解析过,那么后续的引用解析请求就应当一直能成功;同样地,如果第一次解析失败了,其他指令对这个符号的解析请求也应该收到相同的异常,哪怕这个请求的符号在后台已成功加载进虚拟机内存之中;
-
对于invokedynamic指令,上面的规则就不成立了。当碰到某个前面已经由involvedynamic指令触发过解析的符号引用时,并不意味着这个解析结果对于其它invoked dynamic指令也同样生效。因为invoke dynamic指令的目的本来就是用于动态语言支持,它对应的引用称为"动态调用点限定符"。这里的动态的含义时必须等程序时机运行到这条指令时,解析动作才能进行。相对地,其余可触发解析的指令都是“静态”的,可以在刚刚完成加载阶段,还没有开始执行代码时就提前进行解析;
-
解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符这7类符号引用进行,分别对应常量池的CONSTANT_Class_info、CONSTANT_Fieldref_info、CONSTANT_Methodref_info、CONSTANT_InterfaceMethodref_info、CONSTANT_MethodType_info、CONSTANT_MethodHandle_info、CONSTANT_Dynamic_info和CONSTANT_InvokeDynamic_info这8种常量类型;
2.4.2 类或接口的解析
2.4.2.1 解析过程
- 假设当前代码所处的类为D,如果要把一个从未解析过的符号引用N解析为一个类或者接口C的直接引用,那虚拟机完成整个解析的过程需要包括以下3个步骤:
- 如果C不是一个数组类型,那虚拟机将会把代表N的全限定名传递给D的类加载器去加载这个类C。在加载过程中,由于元数据验证、字节码验证的需要,又可能触发其他相关类的加载动作,例如加载这个类的父类或实现的接口。一旦这个加载过程出现了任何异常,解析过程就直接失败;
- 如果C时一个数组类型,并且数组的元素类型为对象,也就是N的描述符会时类似“[Ljava/lang/Integer”的形式,那将会按照第一点的规则加载数组元素类型。如果N的描述符如前面所假设的形式,需要加载的元素类型就是“java.lang.Integer”,接着由虚拟机生成一个代表该数组维度和元素的数组对象;
- 如果上面俩步没有出现任何异常,那么C在虚拟机中实际上已经称为一个有效的类或者接口了,但在解析完成前还要进行符号引用验证,确认D是否具备对C的访问权限。如果发现不具备访问权限,将抛出java.lang.IllegalAccessError异常;
2.4.2.2 模块化
- 针对上面所说的访问权限验证,在JDK9引入了模块化以后,一个public类型也不再意味着程序任何为欸之都有它的访问权限,我们还必须检查模块间的访问权限;
- 如果我们说一个D拥有C的访问权限,那意味着以下3条规则中至少有其中一条成立:
- 被访问类C是public的,并且与访问类D处于同一个模块;
- 被访问类C是public的,不与访问类D处于同一个模块,但是被访问类C的模块允许被访问类D的模块进行访问;
- 被访问类C不是public的,但是他与访问类D处于同一个包中;
2.4.3 字段解析
- 要解析一个从未被解析过的字段符号引用,首先将会对字段表内class_index项中索引的CONSTANT_Class_info符号引用进行解析,也就是字段所属的类或接口的符号引用。如果在解析这个类或接口符号引用的过程中出现了任何异常,都会到这字段符号引用解析的失败。如果解析成功完成,那把这个字段所属的类或接口用C标识,《Java虚拟机规范》要求按照如下步骤对C进行后续字段的搜索:
- 如果C本身就包含了简单每次和字段描述符斗鱼目标相匹配的字段,则返回这个字段的直接引用,查找结束;
- 否则,如果在C中实现了接口,将会按照继承关系从下往上递归搜索各个接口和它的父接口,如果接口中包含了简单名称和字段描述符都与目标相匹配的字段,则返回这个字段的直接引用,查找结束;
- 否则,如果C不是java.lang.Object的话,将会按照继承关系从下往上递归搜索其父类,如果在父类中包含了简单名称和仔段描述符都与目标相匹配的字段,则返回这个字段的直接引用,查找结束;
- 否则,查找失败,抛出java.lang.NoSuchFieldError异常;
- 如果查找过程成功返回了引用,将会对这个字段进行权限验证,如果发现不具备字段的访问权限,将抛出java.lang.IllegalAccessError异常;
- 以上解析规则能确保java虚拟机获得字段唯一的解析结果,但在实际情况中,javac编译器往往会采取比上述规范更加严格一些的约束,譬如有一个同名字段同时出现在某个类的接口和父类当中,或者同时在自己或父类的多个接口中出现,按照解析规则仍是可以确定唯一的访问字段,但javac编译器就可能直接拒绝编译其为Class文件;
2.4.4 方法解析
- 方法解析的第一个步骤与字段解析一样,也是需要先解析出放发表的class_index想中索引的方法所属的类或接口的符号引用,如果解析成功,那么我们依然用C表示这个类,接下来虚拟机将会按照如下步骤进行后续的方法搜索:
- 由于Class文件格式中类的方法和接口的方法符号引用的常量类型定义是分开的,如果在类的方法表中发现class_index中索引的C是个接口的话,那就直接抛出java.lang.IncompatiblaClassChangeError异常;
- 如果通过了第一步,在类C中查找是否有简单名称和描述符都与目标相匹配的方法,如果有则返回这个方法的直接引用,查找结束;
- 否则在类C的父类中递归查找是否有简单名称和描述符都与目标相陪陪的方法,如果有则返回这个方法的直接引用,查找结束;
- 否则,在类C实现的接口列表及它们的父接口之中递归查找是否有简单名称和描述符都与目标相匹配的方法,如果存在匹配的方法,说明类C是一个抽象类,这时候查找结束,抛出java.lang.AbstractMethodEoor异常;
- 否则,宣告方法查找失败,抛出java.lang.NoSuchMethodError;
- 最后,如果查找过程成功返回了直接引用,将会对这个方法进行权限验证,如果发现不具备对此方法的访问权限,将抛出java.lang.IllegalAccessError异常;
2.4.5 接口方法解析
- 接口方法也是需要先解析出接口放发表的class_index项中索引的方法所属的类或接口的符号引用,如果解析成功,依然用C表示这个接口,接下来虚拟机将会按照如下步骤进行后续的接口方法搜索:
- 与类的方法解析相反,如果在接口方法表中发现class_index中的索引C是个类而不是接口,那么就直接抛出java.lang.IncompatibleClassChangeError异常;
- 否则,在接口C中查找是否有简单名称和描述符都与目标相匹配的方法,如果有则返回这个方法的直接引用,查找结束;
- 否则,在接口C的父接口中递归查找,知道java.lang.Object类(接口方法的查找范围也会包括Object类中的方法)为止,看是否有简单名称和描述符都与目标相匹配的方法,如果有则返回这个方法的直接引用,查找结束;
- 对于上述规则,由于java的接口允许多重继承,如果C的不同父类接口中存有多个简单名称和描述符都与目标相匹配的方法,那将会从这多个方法中返回其中一个并结束查找,《java虚拟机规范》中并没有进一步规则约束应该返回哪一个接口方法。但与之前字段查找类似地,不同发行商实现的javac编译器有可能会按照更严格的约束拒绝编译这种代码来避免不确定性;
- 否则,宣告方法查找失败,抛出java.lang.NoSuchMethodError异常;
- 在jdk9一千,java接口中所有方法都是默认public的,也没有模块化的访问约束,所以不存在访问权限的问题,接口方法的符号解析就不可能抛出java.lang.IllegalAccessError异常。但在JDK9中增加了接口的静态私有方法,也有了模块化的访问约束,所以从jdk9起,接口方法的访问也完全有可能因访问权限控制而初夏你java.lang.IllegalAccessError异常;
2.5 初始化
- 类的初始化阶段是类加载过程的最后一个步骤,之前介绍的几个类加载的动作里,除了在加载阶段用户应用程序可以通过自定义类加载器的方式局部参与外,其余东顾总都完全由java虚拟机来进行主导。直到初始化阶段,java虚拟机本身才真正开始开始执行类中编写的java程序代码,将主导权移交给应用程序;
- 进行准备阶段时,变量已经赋值过一次系统要求的初始零值,而在初始化阶段,则会根据程序员通过程序编码制定的主观计划去初始化类变量和其他资源。初始化阶段就是执行类构造器
()方法的过程。 ()并非程序员在Java代码中直接编写的方法,它是java-c编译器的自动生成物,但我们非常由必要了解这个方法具体如何产生的,以及该方法执行过程中各种可能会影响程序运行行为的细节,这部分比起其他类加载过程更贴近普通的程序开发人员的时机工作; ()方法是由编译器自动收集类中的所有类变量的赋值动作和静态语句块(static{}块)中的语句合并产生的,编译器收集的顺序是由语句在源文件中出现的顺序决定的,静态语句块中只能访问到定义在静态语句块之前的变量,定义在它之后的变量,在前面的静态语句块可以赋值,但是不能访问; ()方法与类的构造函数(即在虚拟机视角中的实例构造器 ()方法)不同,它不需要显示地调用父类构造器,Java虚拟机会保证在子类的 ()方法执行前,父类的 ()方法已经执行完毕。因此在Java虚拟机中第一个被执行的 ()方法的类型肯定是java.lang.Object; - 由于父类的
()方法先执行,也就意味着父类中定义的静态语句块要优先于子类的变量的赋值操作; ()方法对于类或者接口来说并不是必需的,如果一个类中没有静态语句块,也没有对变量的赋值操作,那么编译器可以不为这个类生成 ()方法; - 接口中不能使用静态语句块,但仍然由变量初始化的赋值操作,因此接口与类一样都会生成
()方法。但接口与类不同的是,执行接口的 ()方法不需要先执行父接口的 ()方法,因为只有当父接口中定义的变量被使用时,父接口才会被初始化。此外,接口的实现类在初始化时也一样不会执行接口的 ()方法; - java虚拟机必须保证一个类的
()方法在多线程环境中被正确的加锁同步,如果多个线程同时去初始化一个类,那么只会有其中一个线程去执行这个类的 ()方法,其他线程都需要阻塞等待,直到活动线程执行完毕 ()方法。如果一个类的 ()方法中有耗时很长的操作,那就可能造成多个进程阻塞,在实际应用中这种阻塞往往时很隐蔽的;需要注意的时。其他线程虽然会被阻塞,但如果执行 ()方法的那条线程退出 ()方法后,其他线程唤醒后则不会再次进入 ()方法。同一个类加载器下,一个类型只会被初始化一次;

浙公网安备 33010602011771号