3、JVM — 类加载过程

  • 类加载过程
    • 加载
    • 验证
    • 准备
    • 解析
    • 初始化

类加载过程

Class 文件须要加载到虚拟机中以后才能运行和使用,那么虚拟机是如何加载这些 Class 文件呢?html

系统加载 Class 类型的文件主要三步:加载->链接->初始化。链接过程又可分为三步:验证->准备->解析。java

类加载过程

加载

类加载过程的第一步,主要完成下面3件事情:git

  1. 经过全类名获取定义此类的二进制字节流
  2. 将字节流所表明的静态存储结构转换为方法区的运行时数据结构
  3. 在内存中生成一个表明该类的 Class 对象,做为方法区这些数据的访问入口

虚拟机规范多上面这3点并不具体,所以是很是灵活的。好比:"经过全类名获取定义此类的二进制字节流" 并无指明具体从哪里获取、怎样获取。好比:比较常见的就是从 ZIP 包中读取(往后出现的JAR、EAR、WAR格式的基础)、其余文件生成(典型应用就是JSP)等等。github

一个非数组类的加载阶段(加载阶段获取类的二进制字节流的动做)是可控性最强的阶段,这一步咱们能够去完成还能够自定义类加载器去控制字节流的获取方式(重写一个类加载器的 loadClass() 方法)。数组类型不经过类加载器建立,它由 Java 虚拟机直接建立。数组

类加载器、双亲委派模型也是很是重要的知识点,这部份内容会在后面的文章中单独介绍到。安全

加载阶段和链接阶段的部份内容是交叉进行的,加载阶段还没有结束,链接阶段可能就已经开始了。数据结构

验证

验证阶段示意图

准备

准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些内存都将在方法区中分配。对于该阶段有如下几点须要注意:多线程

  1. 这时候进行内存分配的仅包括类变量(static),而不包括实例变量,实例变量会在对象实例化时随着对象一块分配在 Java 堆中。
  2. 这里所设置的初始值"一般状况"下是数据类型默认的零值(如0、0L、null、false等),好比咱们定义了public static int value=111 ,那么 value 变量在准备阶段的初始值就是 0 而不是111(初始化阶段才会复制)。特殊状况:好比给 value 变量加上了 fianl 关键字public static final int value=111 ,那么准备阶段 value 的值就被复制为 111。

基本数据类型的零值:oracle

基本数据类型的零值

解析

解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程。解析动做主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用限定符7类符号引用进行。jvm

符号引用就是一组符号来描述目标,能够是任何字面量。直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。在程序实际运行时,只有符号引用是不够的,举个例子:在程序执行方法时,系统须要明确知道这个方法所在的位置。Java 虚拟机为每一个类都准备了一张方法表来存放类中全部的方法。当须要调用一个类的方法的时候,只要知道这个方法在方发表中的偏移量就能够直接调用该方法了。经过解析操做符号引用就能够直接转变为目标方法在类中方法表的位置,从而使得方法能够被调用。

综上,解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程,也就是获得类或者字段、方法在内存中的指针或者偏移量。

初始化

初始化是类加载的最后一步,也是真正执行类中定义的 Java 程序代码(字节码),初始化阶段是执行类构造器 <clinit> ()方法的过程。

对于<clinit>() 方法的调用,虚拟机会本身确保其在多线程环境中的安全性。由于 <clinit>() 方法是带锁线程安全,因此在多线程环境下进行类初始化的话可能会引发死锁,而且这种死锁很难被发现。

对于初始化阶段,虚拟机严格规范了有且只有5中状况下,必须对类进行初始化:

  1. 当遇到 new 、 getstatic、putstatic或invokestatic 这4条直接码指令时,好比 new 一个类,读取一个静态字段(未被 final 修饰)、或调用一个类的静态方法时。
  2. 使用 java.lang.reflect 包的方法对类进行反射调用时 ,若是类没初始化,须要触发其初始化。
  3. 初始化一个类,若是其父类还未初始化,则先触发该父类的初始化。
  4. 当虚拟机启动时,用户须要定义一个要执行的主类 (包含 main 方法的那个类),虚拟机会先初始化这个类。
  5. 当使用 JDK1.7 的动态动态语言时,若是一个 MethodHandle 实例的最后解析结构为 REF_getStatic、REF_putStatic、REF_invokeStatic、的方法句柄,而且这个句柄没有初始化,则须要先触发器初始化。

参考

相关文章
相关标签/搜索