程序员公众号【麦小丁】征集优质文章

虚拟机类加载机制

2018-12-12  本文已影响21人  小螺钉12138

1、类加载时机

类从被加载到虚拟机内存中开始,到卸载出内存为止,它的整个生命周期包括:加载、验证、准备、解析、初始化、使用和卸载7个阶段。

graph LR
A[加载]-->B[验证]
B-->C[准备]
C-->D[解析]
D-->E[初始化]
E-->F[使用]
F-->G[卸载]

加载、验证、准备、初始化和卸载这5个阶段的顺序是确定的,只有解析阶段在某些情况下可以在初始化阶段之后开始,这是为了支持Java语言的运行时绑定(也称为动态绑定或晚期绑定)。

虚拟机加载阶段在Java虚拟机规范中并没有进行强制约束,由虚拟机的具体实现来自由把握。虚拟机规范严格规定了有且只有5种情况必须立即对类进行“初始化”(加载、验证、准备自然在此之前开始):

有且只有这5种场景中的行为称为对一个类进行主动引用。除此之外,所有引用类的方式都不会触发其初始化,称为被动引用。例下面三种引用不会导致初始化:

接口和类初始化有所区别的地方在于:当一个类初始化时,要求其父类全部都已经初始化过了,但是接口初始化时,并不要求其父接口全部完成初始化,只有在使用到父接口时(引用父接口中的常量)才会初始化

2、类加载过程

2.1、加载

在加载阶段,虚拟机需要完成以下3件事情:

获取二进制字节流的途径:

相对于类加载过程的其他阶段,一个非数组的加载阶段(加载阶段中获取类的二进制字节流的动作)是开发人员可控性最强的,因为加载阶段既可以使用系统提供的引导类加载器来完成,也可以由用户自定义的类加载器去完成(开发人员通过定义自己的类加载器去控制字节流的获取方式,即重写一个类加载器的loadClass()方法)。

对于数组类而言,情况就有所不同,数组类本身不通过类加载器创建,它是由Java虚拟机直接创建的。但数组类与类加载器仍然有很密切的关系,因为数组类的元素类型(Element Typ
e,指的是数组去掉所有维度的类型)最终是要靠类加载器去创建,一个数组类创建过程遵循以下规则:

加载阶段完成之后,虚拟机外部的二进制字节流就按照虚拟机所需的格式存储在方法区之中,方法区中的数据存储格式由虚拟机实现自行定义,虚拟机规范未规定此区域的具体数据结构。然后在内存中实例化一个java.lang.Class类的对象(并没有明确规定是在Java堆中,对于HotSpot虚拟机而言,Class对象比较特殊,它虽然是对象,但是存放在方法区里面),这个对象将作为程序访问方法区中的这些类型数据的外部接口。

加载阶段与连接阶段的部分内容是交叉进行的,加载阶段尚未完成,连接阶段可能已经开始,但是这两个阶段的开始时间仍然保持固定的先后顺序。

2.2、验证

验证是连接阶段的第一步,这一阶段目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。

2.3、准备

准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些变量所使用的内存都将在方法区中进行分配。这个阶段中有两个容易混淆的概念,首先,这时候进行内存分配的仅包括类变量(被static修饰的变量),而不包括实例变量,实例变量将会在对象实例化随着对象一起分配在Java堆中。其次,这里所说的初始值“通常情况”下是数据类型的零值,类变量的赋值动作是在初始化阶段才会执行。

2.4、解析

解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。

解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符7类符号引用进行。

1、类或接口的解析
假设当前代码所处的类为D,如果要把一个从未解析过的符号引用N解析为一个类或接口C的直接引用,那么虚拟机完成整个解析过程需要3个步骤:

2、字段解析
虚拟机规范要求按照如下步骤对C进行后续字段的搜索

查找过程成功返回引用,将会对这个字段进行权限验证,如果发现不具备对字段的访问权限,将抛出java.lang.IlleglAccessError异常。

3、类方法解析
类方法解析第一个步骤与字段解析一样,也需要先解析出类方法表的class_index项中索引的方法所属的类或接口的符号引用,如果解析成功,我们依然用C表示这个类,接下来虚拟机将会按照如下步骤进行后续的类方法搜索:

4、接口方法解析
接口方法也需要先解析出接口方法表的class_index项中索引的方法所属的类或接口的符号引用,如果解析成功,依然用C表示这个接口,后续接口方法搜索步骤:

2.5初始化

类初始化阶段是类加载阶段过程的最后一步,前面的类加载过程中,除了在加载阶段用户应用程序可以通过自定义类加载器参与之外,其余动作完全由虚拟机主导和控制。到了初始化阶段,才真正开始执行类中定义的Java程序代码(或者说是字节码)。

初始化阶段是执行类构造器<clinit>()方法的过程。<clinit>()方法执行过程中一些可能会影响程序运行行为的特点和细节

3、类加载器

虚拟机团队把类加载阶段中的“通过一个类的全限定名来获取描述此类的二进制字节流”这个动作放到Java虚拟机外部去实现,以便让应用程序自己决定如何去获取所需要的类。实现这个动作的代码模块成为“类加载器”。
类加载器在类层次划分、OSGi、热部署、代码加密等领域大放异彩,成为了Java技术体系的一块基石。

3.1、类与类加载器

比较两个类是否“相等”,只有在这两个类是由同一个类加载器加载的前提下才有的意义,否则,即使这两个类来源于同一个class文件,被同一个虚拟机加载,只要加载它们的类加载器不同,那么这两个类就必定不相等。

3.2、双亲委派模型

从Java虚拟机的角度来讲,只存在两种不同的类加载器:一种是启动类加载器(Bootstrap ClassLoader),这个类加载器由C++实现,是虚拟机自身的一部分;另一种就是所有其他的类加载器,这些类加载器都由Java语言实现,独立于虚拟机外部,并且全部都继承自抽象类java.lang.ClassLoader。

从开发人员的角度划分,类加载器还可以划分得更细致一些,绝大多数Java程序都会使用以下3种系统提供的类加载器。
-1、启动类加载器:这个类加载器负责将存放在<JAVA_HOME>\lib目录中的,或者被-Xbootclasspath参数所指定的路径中的,并且是虚拟机识别的(仅按照文件名识别,如rt.jar)类库加载到虚拟机内存中。启动类加载器无法被Java程序直接引用,用户在编写自定义类加载器时,如果需要把加载请求委派给引导类加载器,那直接使用null代替即可。

使用双亲委派模型来组织类加载器的优点在于:一个类java.lang.Object,存放在rt.jar中无论哪一个类加载器要加载这个类,最终都是委派给处于模型

上一篇 下一篇

猜你喜欢

热点阅读