深入理解JVM(七)JVM类加载机制

7.1JVM类加载机制

  虚拟机把数据从Class文件加载到内存,并且校验、转换解析和初始化最终形成可以被虚拟机使用的Java类型,这就是虚拟机的类加载机制。

7.2类加载的时机

  1.类加载的步骤开始的顺序: 加载(Loading) -> 验证(Verification) -> 准备(Preparation) -> 解析(Resolution) -> 初始化(Initialization) -> 使用(Using) -> 卸载(Unloading) ,验证、准备、解析的过程称为 链接 ,而加载、验证、准备、初始化和卸载的执行的开始顺序是确定的,而解析可能在初始化之后开始;

  2.关于初始化阶段,有5种情况需要立即进行初始化:

    (1)遇到这四个字节码指令时:new、getstatic、putstatic、invokestatic,如果类未进行过初始化,那么进行初始化,这几个字节码指令所在场景:new对象、调用类的静态属性、调用类的静态方法;

    (2)使用 java.lang.reflect 包的方法对类进行反射调用时,如果未进行过初始化,那么进行初始化;

    (3)当初始化一个类时,其父类未进行初始化时,对其父类进行初始化;

    (4)虚拟机启动时需要初始化一个指定的主类(包含main()方法的类);

    (5)使用jdk1.7, java.lang.invoke.MethodHandle 实例的解析结果为 REF_getStatic 、 REF_putStatic 、 REF_invokeStatic 的方法句柄,如果此句柄对应的类未初始化,那么进行初始化。

  3.只有以上五种情况类才会被初始化,它们也叫 主动引用 ,而其他的引用类的方式则不会初始化类,它们叫做 被动引用 :

    (1)使用子类访问父类的静态属性时,不会初始化子类;

    (2)创建一个类的数组时,不会初始化此类,但是会初始化出一个另外的类,代表这个数组对象;

    (3)访问一个类的静态常量时,不会初始化这个类,这个静态常量进入常量池会被归属给NonInitialization类的常量池中,代表不会初始化;

7.3类加载的过程

  1.加载:

    (1)通过类的全限定名来获取Class文件的二进制流;

    (2)将这个字节流根据虚拟机所需要的存储结构存放在内存中;

    (3)生成这个类的 java.lang.Class 对象,作为访问类的数据的外部接口;

  2.验证:

    虽然Java代码的编译过程不允许一些不安全的做法(C/C++常做),比如:访问数组边界以外的数据、错误的对象转型、跳转到不存在的行,但是不能保证Class文件不被修改,所以验证这一步骤作为连接的第一个阶段对于JVM的安全性来说非常重要;

    验证的过程又分为四个阶段: 文件格式验证 、 元数据验证 、 字节码验证 、 符号引用验证

    (1)文件格式验证:①验证魔数;②主、次版本号;③长度检查;等等

    (2)元数据验证:①验证除 java.lang.Object 类以外其他类有无继承父类;②是否继承了final类;③非抽象类是否重写了必须重写的方法;④字段、方法和父类是否矛盾;等等

    (3)字节码验证:①验证错误的对象转型;②跳转到不存在的行;等等

    (4)符号引用验证:①验证能不能通过符号引用的类的全限定名找到这个类;②验证这个符号引用的类、字段和方法的可访问性(public、private);等等,验证完成之后,则符号引用转化为 直接引用 (内存地址引用);

  3.准备:

     准备阶段是为类变量(static修饰)分配内存并赋予初始值的阶段;

    (1)这里说的赋予初始值说的一般都是零值,比如: public static int i = 1; 这里的 i 在准备阶段完成之后会被赋值为0而非1,赋值为1那是初始化阶段;

    (2) public static final int i = 1; 而这里的 i在准备阶段之后赋值为1;

  4.解析:

    将常量池内的 符号引用 转换为 直接引用 的过程,分为 类或接口解析 、 字段解析 、 类方法解析 、 接口方法解析 。

  5.初始化:

    初始化阶段才算是真正开始运行Java代码,初始化阶段就是运行类构造器的 <cinit>() 方法,对应 static{} 方法块;

    (1)<cinit>()方法只能访问到static{}代码块前面的变量,而在static后面的变量,只能赋值,不能访问引用(非法向前引用);

    (2)子类初始化,会默认先初始化父类来调用父类的<cinit>()方法;

    (3)一般类中没有static{}代码块,那么也就不会生成<cinit>()方法;

    (4)接口不能写static{}代码块,但是赋值给变量时也会生成<cinit>()方法;

    (5)<cinit>()方法是同步的,线程安全的。

7.4类加载器

   类加载器 作用于 加载 阶段中,根据类的全限定名来获取Class文件的二进制流。

  1.关于类和类加载器:

    两个类要相等( equals() ),它们首先要是同一个类加载器进行加载的;

  2.双亲委派模型:

    (1)三种系统的类加载器:

      ①启动类加载器(Bootstrap ClassLoader):加载 %JAVA_HOME%/lib 下和 -Xbootclasspath 指定的目录下的类库;

      ②扩展类加载器(Extension ClassLoader):加载 %JAVA_HOME%/lib/ext 目录下和 java.ext.dirs 系统变量所指定的目录下的类库;

      ③ 应用程序类加载器(Application ClassLoader):加载用户类路径下的类库;

    (2)双亲委派模型:

      如下图所示,要求除 启动类加载器(Bootstrap ClassLoader) 以外,其它类加载器都要有自己的父加载器,它们之间不是 继承 关系,而是 组合 复用的关系;

      

    (3)双亲委派机制工作过程:

      一个类加载器在收到类加载的请求时,不会立即去加载这个类,而是向 父类加载器 请求加载,依次类推,直到顶层类加载器,只有当类加载器不能加载此类时才会让 子类加载器 去加载这个类;

    (4)双亲委派机制的意义:

      如此保证了类不会因为不同类加载器导致加载出不同的类,从而使程序混乱,例如自己写一个 java.lang.String 类,系统只加载了jdk默认的 java.lang.String 的类文件,而不会加载自己写的java.lang.String类;

  3.破坏双亲委派模型:

原文地址:https://www.cnblogs.com/lcmlyj/p/10180721.html

时间: 2024-10-06 18:08:33

深入理解JVM(七)JVM类加载机制的相关文章

深入理解和探究Java类加载机制-

深入理解和探究Java类加载机制---- 1.java.lang.ClassLoader类介绍 java.lang.ClassLoader类的基本职责就是根据一个指定的类的名称,找到或者生成其对应的字节代码,然后从这些字节代码中定义出一个Java 类,即 java.lang.Class类的一个实例. ClassLoader提供了一系列的方法,比较重要的方法如: 2.JVM中类加载器的树状层次结构 Java 中的类加载器大致可以分成两类,一类是系统提供的,另外一类则是由 Java 应用开发人员编写

【深入理解JVM】:类加载机制

概述 虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 与那些在编译时需要进行链接工作的语言不同,在Java语言里,类型的加载.连接和初始化过程都是在程序运行期间完成的,例如import java.util.*下面包含很多类,但是,在程序运行的时候,虚拟机只会加载哪些我们程序需要的类.这种策略虽然会令类加载时稍微增加一些性能开销,但是会为Java应用程序提供高度的灵活性. 类加载的时机 类从

图解JVM和Tomcat类加载机制

说到本篇的tomcat类加载机制,不得不说翻译学习tomcat的初衷. 之前实习的时候学习javaMelody的源码,但是它是一个Maven的项目,与我们自己的web项目整合后无法直接断点调试.后来同事指导,说是直接把java类复制到src下就可以了.很纳闷....为什么会优先加载src下的java文件(编译出的class),而不是jar包中的class呢? 现在了解tomcat的类加载机制,原来一切是这么的简单. 类加载 在JVM中并不是一次性把所有的文件都加载到,而是一步一步的,按照需要来加

【JVM】虚拟机类加载机制

什么是类加载 虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. <[JVM]类文件结构>讲的是Class文件结构,即我们编写的Java代码(.java文件)经过编译后生成Class文件(.class文件).这一章讲述的是如何将这个Class文件加载到内存并最终形成虚拟机直接使用Java类型的过程. 1.类加载的时机 类的生命周期 类的生命周期 其中,加载.验证.准备.初始化和卸载这5个顺序

jvm之java类加载机制和类加载器(ClassLoader)的详解

当程序主动使用某个类时,如果该类还未被加载到内存中,则JVM会通过加载.连接.初始化3个步骤来对该类进行初始化.如果没有意外,JVM将会连续完成3个步骤,所以有时也把这个3个步骤统称为类加载或类初始化. 一.类加载过程 1.加载 加载指的是将类的class文件读入到内存,并为之创建一个java.lang.Class对象,也就是说,当程序中使用任何类时,系统都会为之建立一个java.lang.Class对象. 类的加载由类加载器完成,类加载器通常由JVM提供,这些类加载器也是前面所有程序运行的基础

【深入理解Java虚拟机】类加载机制

本文内容来源于<深入理解Java虚拟机>一书,非常推荐大家去看一下这本书. 本系列其他文章: [深入理解Java虚拟机]Java内存区域模型.对象创建过程.常见OOM [深入理解Java虚拟机]垃圾回收机制 1.类加载机制概述 虚拟机把描述类的数据从Class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 在java中,类型的加载.连接和初始化过程都是在程序运行期间完成的,这种策略虽然会带来一些性能开销,但是却为jav

深入理解Java虚拟机-----------虚拟机类加载机制

虚拟机类加载机制 类从被加载到虚拟机内存开始,到卸载出内存为止,整个生命周期包括:加载,验证,准备,解析,初始化,使用,卸载等7个阶段.其中,验证,准备,解析3个部分称为连接. 以上7个阶段中,加载,验证,准备,初始化和卸载五个阶段的顺序是确定的,类的加载过程必需按照这种顺序按部就班的开始(开始并不意味着按部就班的"进行"或"完成",因为这些阶段通常是互相交叉地混合式进行的).而解析阶段则不一定:它在某些情况下可以在初始化阶段之后再开始,这是为了支持Java语言的运

JVM:虚拟机类加载机制

Java语言,类型的加载.连接.初始化都是在程序运行期间完成的 类的生命周期:加载(Loading).验证(Verification).准备(Preparation).解析(Resolution).初始化(Initialization).使用(Using).卸载(Unloading) 连接(Linking):验证.准备.解析 加载.验证.准备.初始化.卸载这5个阶段的顺序是确定的 有且仅有5种情况必须对类进行"初始化" 遇到new.getStatic.putStatic.ivokeSt

Java核心技术梳理-类加载机制与反射

一.引言 反射机制是一个非常好用的机制,C#和Java中都有反射,反射机制简单来说就是在程序运行状态时,对于任意一个类,能够知道这个类的所有属性和方法,对于任意一个对象,能够调用它的任意属性和方法,其实初听就知道反射是一个比较暴力的机制,它可能会破坏封装性. 通过反射的定义我们可以想到反射的好处:可以灵活的编写代码,代码可以在运行时装配,降低代码的耦合度,动态代理的实现也离不开反射. 为了更好的理解反射,我们先了解下JVM中的类加载机制. 二.类加载机制 当程序要使用某个类时,如果这个类还未加载