JVM之类加载机制

2018-07-08  本文已影响0人  Lebens

一个Java类从被加载到虚拟机内存到被卸载出内存为止,生命周期一共包括如下几个阶段:

其中验证、准备、解析这个3个部分统称为链接(Linking)。

类的生命周期图.png

加载、验证、准备、初始化和卸载这5个阶段开始执行的顺序是一定的,但不意味着这几个阶段是分开执行的,这些阶段通常是相互交叉混合式进行的,通常会在一个阶段执行的过程中调用、激活另一个过程。

解析阶段则不一定,为了支持java的运行时绑定,在某些特定的情况下解析可以在初始化阶段之后开始。

加载阶段

这个阶段主要完成如下3件事情

这里获取字节流的方式并不局限于zip,还包括诸如网络中获取、运行时生成、其他文件生成、数据库读取等方式。

同时相对于类加载过程的其他阶段,相对于一个非数组类的加载阶段是开发者可控性最强的,因为加载阶段既可以使用系统提供的加载器,也可以用户自定义类加载器来完成类的加载。

数组类的加载情况有所不同,虽然数组类是JVM直接创建的,但是数组的组件,最终还是要依靠类加载器去加载,一个数组类创建主要有如下几点:

加载阶段完成后,类的二进制字节流将按照JVM所需的格式存储在方法区中,同时在内存中实例化一个java.lang.Class的实例对象,作为程序访问方法区中这些类数据的外部接口。相对于HotSpot,这个实例对象比较特殊,虽然是一个对象,但并没有放置在堆中,而是放置在方法区中

验证阶段

这个阶段主要是确保加载的Class文件中的字节流包含的信息符合当前虚拟机的要求,同时不存在损害JVM的行为.

从整体上看,验证阶段大致上会完成下面4个阶段的检验动作:文件格式验证、元数据验证、字节码验证、符号引用验证

文件格式验证

这一阶段主要验证字节流是否符合Class文件格式的规范:

以及等等一些步骤,通过这个步骤的验证之后,字节流才会进入方法区中存储,后续的3个阶段的验证,都是基于方法区的存储结构进行的

元数据验证

这一阶段主要是对字节码描述的信息进行语义分析:

等等

字节码验证

这一阶段将对类的方法体进行校验分析,保证类的方法在运行时不会做出危害虚拟机安全的事件。这个阶段是整个验证过程中最复杂的一个阶段

等等

没有通过字节码验证的方法体一定是有问题的,但是通过字节码验证的方法体也不能保证一定没问题

符号引用验证

这个验证发生在JVM将符合引用转换成直接引用的时候,在链接的第三阶段---解析阶段发生

等等

验证阶段对于JVM的类加载机制是非常重要的,但不是必要的,因为对程序运行期没有影响

准备阶段

这个阶段会执行类的<clint>(),主要是为类变量(也就是被static修饰的)分配内存并赋值零值的阶段。所谓的零值就是默认的初始值,每种数据类型各有不同的零值(需要注意的是同时被static final 修饰的变量在此时就赋值完毕,并不会存在赋零值的操作)。类的<clint>是编译器自动收集类变量以及静态代码块后自动合并生成的。

举个栗子:

public static int abc = 123;
public static final int ABC = 123;

上面两句的代码在准备阶段的区别是

解析阶段

这个阶段主要是将字节码文件中的符号引用转化为直接引用,这个阶段发生的时间段并不确定,某些情况下解析可以发生在初始化阶段之后,这是为了支持Java语言的运行时绑定。JVM可以根据需求来判断到底要在类被加载器加载的时候就对常量池中的符号引用进行解析,还是等到另一个符号引用将要被使用的时候去解析。

解析主要针对类或接口、字段、类方法、类接口、接口方法、方法类型、方法句柄和调用点限定符7种符号引用,这里主要讲前面4种。

类或接口的解析

假设在类D中要把一个未解析过的符号引用N解析为对一个类或接口C的直接引用,JVM有如下操作:

字段解析

解析一个字段符号引用,会先对字段所属的类或者接口的符号引用先进行解析。解析到所属的类或接口C后JVM按照如下规范对字段进行搜索:

类方法解析

与字段解析一样,解析一个类方法也是要先解析到这个方法所属的类或者接口。解析到所属的类或接口C后JVM按照如下规范对字段进行搜索:

接口方法解析

与类方法解析相比较:

初始化阶段

类的初始化是类加载过程的最后一步,到了初始化阶段才开始真正执行java代码

编译器自动收集实例变量初始化以及实例代码块后自动合并生成类的<init>()

子类初始化时会先调用父类<init>(),用以保证子类能正常初始化。
执行子类的<init>()

使用阶段

这个没啥好说的,只要代表类的Class对象还能被引用到,类就还在使用当中。

卸载阶段

一个类何时结束生命周期,取决于代表它的Class对象何时结束生命周期。类的卸载,其实就是方法区中的类的回收。判断一个类是否可以进行回收,需要同时满足下面3个条件:

上面条件仅仅说明该类可以进行回收,但是并不想类的实例对象一样,不使用了就立马进行回收

类回收示意图.png

如上图所示,当左侧所有的引用都消失时,类可以被回收。

Java虚拟机自带的类加载器包括根类加载器、扩展类加载器和系统类加载器。由Java虚拟机自带的类加载器所加载的类,在虚拟机的生命周期中,始终不会被卸载。Java虚拟机本身会始终引用这些类加载器,而这些类加载器则会始终引用它们所加载的类的Class对象,因此这些Class对象始终是可触及的。由用户自定义的类加载器加载的类是可以被卸载的。

参考书籍

本文摘录、整理自周志明的《深入理解Java虚拟机》一书,如想获得更详细介绍可自己查阅此书。

上一篇 下一篇

猜你喜欢

热点阅读