我们在编写Java程序之后,会通过编译器得到一个class文件,这个class文件是如何与JVM进行配合的呢?类中的信息是如何变成JVM可以使用的Java类型呢?这些都是类加载机制做到的。

虚拟机把描述类的数据从class文件中加载到内存里,并对数据进行校验,转换解析和初始化,最终形成被虚拟机直接使用的Java类型,这就是类加载机制。

类的生命周期

一个类从加载进入内存到卸载出内存,一共经过以下几个生命周期。

  1. 加载
  2. 验证
  3. 准备
  4. 解析
  5. 初始化
  6. 使用
  7. 卸载

这篇文章主要说明前5个生命周期,也就是说主要总结类从加载到使用的过程。

类的加载的过程

类的第一个生命周期是加载,那么类是什么时候执行加载过程的呢?这个过程JVM规范并没有明确说明,但是针对类的初始化,JVM给出了硬性的规定,只有以下五种情况才会进行类的初始化。

  • 遇到new,getStaic,putStatic,invokestatic这四个字节码指令时,这四个字节码对应的Java代码场景是:使用new关键字实例化对象,读取或设置一个类的静态属性的时候(该属性被final修饰除外),调用一个类的静态方法的时候。
  • 使用反射对类进行调用的时候
  • 当初始化一个类,而他父类还没有初始化的时候,则需要先初始化他的父类
  • main函数所在的那个类。
  • JDK7中一个methodHandle实例最后的解析结果为REF_getStatic,REF_putStatic,REF_invokeStatic,该方法对对应的类还没有初始化的时候,就会触发其初始化。该方法详见博客

上述说明了类初始化的时机,那么类的加载自然要在初始化之前。这也说明了在运行时才会执行类的初始化。
那么加载的这个过程,JVM做了什么工作呢?在加载阶段,JVM主要完成3件事情

  1. 通过一个类的全限定名来获取定义此类的二进制字节流
  2. 将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构
  3. 在内存中生成一个代表这个类的java.lang.Class对象,作为方法区这个类的各种数据的访问入口。

我们先说第一步,通过一个类的全限定名来获取此类的二进制字节流。这个说法实际上是很大的一个操作的空间,这也是各种Java技术能够实现的基础。比如,我们可以通过zip包中获取,那么就有了后来的jar,war,我们还可以通过网络中获取这个类,这就是applet,甚至动态代理,运行时在生成特定的class二进制流,或者由其他文件生成,比如JSP,由JSP文件生成对应的Class文件。

那么,为什么这一步会出现这么多种类繁多的加载技术呢,是因为通过一个类的全限定名来获取定义此类的二进制字节流这一动作是放到了Java虚拟机外部去实现的,是为了方便让应用自己去决定如何获取所需要的类。实现这个动作的功能是常说的类加载器。

类加载器的功能就是将获取class文件,并将其转换为class对象。因为class对象是由类加载器得到的,所以如果比较两个类是否相等,也必须在同一个类加载器加载的前提下,这里的相等时包括class对象的equals,instanceOf等作出的判断。

那么,针对JDK中这么多的类,以及还有我们自己编写的类,是否都是用同一个类加载器加载的呢?这显然不是,我们从Java开发者角度来看看究竟有哪些类加载器,他们是怎么配合的。

  • 启动类加载器
  • 扩展类加载器
  • 应用程序类加载器

启动类加载器是加载%JAVA_HOME/lib目录中的类,虚拟机按照名字来识别,比如rt.jar,不是虚拟机认可的名字,即使放在下面也没有用。该加载器是JVM自身的一部分。

扩展类加载器是负责加载$JAVA_HOME/lib/ext目录下的类。该加载器继承自ClassLoader。

应用程序加载器或者叫做系统类加载器,负责加载ClassPath上指定的类,如果应用程序没有自定义过自己的类加载器,一般情况这个就是程序默认的类加载器。

这三种加载器是相互配合使用,配合的方式是双委派模型,如下图所示:

双亲委派模型的工作过程如下:
如果一个类加载器收到了类加载的过程,他首先不会自己去尝试加载这个类,而是把这个类委派给自己的父类加载器去完成,每一个层次的加载器都是如此,因为所有的类的加载请求都会最终传到顶层的启动类加载器中,只有父类的加载器无法完成这个加载请求时,子加载器才会去尝试自己的加载。
这样做的做的目的在于,保持类在整个JVM中唯一性,上面我们说过了,只有同一个类加载器加载出的类才是相等,如果我们编写了一个Object类,而不采用双亲委派模型的形式去加载,那么会加载出很多不同的Object类。

至此,类的加载阶段就介绍完毕,告一段落了。

类的链接阶段

类的链接阶段主要包含验证,准备,解析这三个阶段,本文简要说明这个三个阶段做的事情。

  • 验证阶段是验证加载类的信息是否会危害虚拟机
  • 准备阶段是正式将类变量分配内存并设置类初始值的阶段,这些都是在方法区进行分配
  • 解析阶段是将常量池的符号引用替换为直接引用的过程

类的初始阶段

类的初始化阶段才会真正执行Java代码中那些语句。

初始化阶段是执行类构造器<clinit>()方法的过程。类构造器<clinit>()方法是由编译器自动收藏类中的所有类变量的赋值动作和静态语句块(static块)中的语句合并产生
当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化
虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确加锁和同步

理解JVM——类加载机制的更多相关文章

  1. 深入理解JVM虚拟机6:深入理解JVM类加载机制

    深入理解JVM类加载机制 简述:虚拟机把描述类的数据从class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 下面我们具体 ...

  2. 深入理解JVM类加载机制

    1.什么是类加载机制? JVM把class文件加载到内存里面,并对数据进行验证.准备.解析和初始化,最终能够被形成被JVM可以直接使用的Java类型的过程. 生命周期包含:加载,验证,准备,解析,初始 ...

  3. 深入理解JVM类加载机制 classloader

    转自https://www.cnblogs.com/ygj0930/p/6536048.html

  4. 理解Java类加载机制(译文)

    理解java类加载机制 你想写类加载器?或者你遇到了ClassCastException异常,或者你遇到了奇怪的LinkageError状态约束异常.应该仔细看看java类的加载处理了. 什么是类加载 ...

  5. JVM基础系列第7讲:JVM 类加载机制

    当 Java 虚拟机将 Java 源码编译为字节码之后,虚拟机便可以将字节码读取进内存,从而进行解析.运行等整个过程,这个过程我们叫:Java 虚拟机的类加载机制.JVM 虚拟机执行 class 字节 ...

  6. Java虚拟机(四):JVM类加载机制

    1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构 ...

  7. JVM类加载机制(转)

    原文出自:http://www.cnblogs.com/ityouknow/p/5603287.html 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运 ...

  8. JVM类加载机制详解

    引言 如下图所示,JVM类加载机制分为五个部分:加载,验证,准备,解析,初始化,下面我们就分别来看一下这五个过程. 加载 在加载阶段,虚拟机需要完成以下三件事情: 1)通过一个类的全限定名来获取定义此 ...

  9. 一夜搞懂 | JVM 类加载机制

    前言 本文已经收录到我的Github个人博客,欢迎大佬们光临寒舍: 我的GIthub博客 学习导图 一.为什么要学习类加载机制? 今天想跟大家唠嗑唠嗑Java的类加载机制,这是Java的一个很重要的创 ...

随机推荐

  1. Java Swing实战(二)下拉菜单组件JComboBox及其事件监听

    接下来给"数据源配置"面板添加下拉框. /** * @author: lishuai * @date: 2018/11/26 13:51 */ public class Weimi ...

  2. 最全Vue开发环境搭建

    前言 一直想去学Vue,不过一直找不到一个契机.然公司手机端用到了跨平台开发apicloud,里边涉及到Vue组件化开发,例如header和footer的封装,以及apicloud自定义的frame等 ...

  3. Zookeeper学习文章目录1

    目录:参考文章如下 1.ZooKeeper学习第一期---Zookeeper简单介绍 2. ZooKeeper学习第二期--ZooKeeper安装配置 3. ZooKeeper学习第三期---Zook ...

  4. Hello world &博客客户端试用

    第一篇博客,使用 open live writer客户端进行测试,下载地址见http://openlivewriter.org/,软件为英文,但配置比较简单,选择“其他博客类型”就ok. 同时安装了语 ...

  5. swiper h5学习

    http://www.swiper.com.cn/ 较多用于移动端

  6. redis中文

    Redis 是完全开源免费的,遵守BSD协议,先进的key - value持久化产品.它通常被称为数据结构服务器,因为值(value)可以是         字符串(String),         ...

  7. JVM知识(三):内存模型和可见性

    这篇文章我们将根据JVM的内存模型探索java当中变量的可见性以及不同的java指令在并发时可能发生的指令重排序的情况.来聊聊java线程对一个变量的更新怎么通知另一个线程,及volatile的作用和 ...

  8. react实例:理解dva构建项目的原理

    请点击相应的步骤查看详情 我们首先搭建一个 dva Demo  项目(请参考react快速构建一个应用项目),然后逐步完成以下内容: 结构划分 设计 Model 组件设计方法 组件设计实践 添加 Re ...

  9. save与Update的合并操作 标签: 关系映射 2017-07-13 15:11 7人阅读 评论(0) 收藏

    做save与update的方法合并操作时,判断条件是主体对象的ID是否存在. 但是当页面中,涉及到多个主体对象的关联对象时,情况变得复杂起来,特总结项目中的几点 一.页面中的VO对象属性可以分为三类: ...

  10. TestNG 判断文件下载成功

    用WatchService写一个方法放在onTestStart()方法里监听文件夹的变化. 但是判断下载成功还需要写一个方法, 用来判断什么时候文件从.xlsx.rcdownload改成.xlsx才行 ...