理解JVM——类加载机制
我们在编写Java程序之后,会通过编译器得到一个class文件,这个class文件是如何与JVM进行配合的呢?类中的信息是如何变成JVM可以使用的Java类型呢?这些都是类加载机制做到的。
虚拟机把描述类的数据从class文件中加载到内存里,并对数据进行校验,转换解析和初始化,最终形成被虚拟机直接使用的Java类型,这就是类加载机制。
类的生命周期
一个类从加载进入内存到卸载出内存,一共经过以下几个生命周期。
- 加载
- 验证
- 准备
- 解析
- 初始化
- 使用
- 卸载
这篇文章主要说明前5个生命周期,也就是说主要总结类从加载到使用的过程。
类的加载的过程
类的第一个生命周期是加载,那么类是什么时候执行加载过程的呢?这个过程JVM规范并没有明确说明,但是针对类的初始化,JVM给出了硬性的规定,只有以下五种情况才会进行类的初始化。
- 遇到
new,getStaic,putStatic,invokestatic这四个字节码指令时,这四个字节码对应的Java代码场景是:使用new关键字实例化对象,读取或设置一个类的静态属性的时候(该属性被final修饰除外),调用一个类的静态方法的时候。 - 使用反射对类进行调用的时候
- 当初始化一个类,而他父类还没有初始化的时候,则需要先初始化他的父类
- main函数所在的那个类。
- JDK7中一个methodHandle实例最后的解析结果为
REF_getStatic,REF_putStatic,REF_invokeStatic,该方法对对应的类还没有初始化的时候,就会触发其初始化。该方法详见博客
上述说明了类初始化的时机,那么类的加载自然要在初始化之前。这也说明了在运行时才会执行类的初始化。
那么加载的这个过程,JVM做了什么工作呢?在加载阶段,JVM主要完成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——类加载机制的更多相关文章
- 深入理解JVM虚拟机6:深入理解JVM类加载机制
深入理解JVM类加载机制 简述:虚拟机把描述类的数据从class文件加载到内存,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类加载机制. 下面我们具体 ...
- 深入理解JVM类加载机制
1.什么是类加载机制? JVM把class文件加载到内存里面,并对数据进行验证.准备.解析和初始化,最终能够被形成被JVM可以直接使用的Java类型的过程. 生命周期包含:加载,验证,准备,解析,初始 ...
- 深入理解JVM类加载机制 classloader
转自https://www.cnblogs.com/ygj0930/p/6536048.html
- 理解Java类加载机制(译文)
理解java类加载机制 你想写类加载器?或者你遇到了ClassCastException异常,或者你遇到了奇怪的LinkageError状态约束异常.应该仔细看看java类的加载处理了. 什么是类加载 ...
- JVM基础系列第7讲:JVM 类加载机制
当 Java 虚拟机将 Java 源码编译为字节码之后,虚拟机便可以将字节码读取进内存,从而进行解析.运行等整个过程,这个过程我们叫:Java 虚拟机的类加载机制.JVM 虚拟机执行 class 字节 ...
- Java虚拟机(四):JVM类加载机制
1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构 ...
- JVM类加载机制(转)
原文出自:http://www.cnblogs.com/ityouknow/p/5603287.html 1.什么是类的加载 类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运 ...
- JVM类加载机制详解
引言 如下图所示,JVM类加载机制分为五个部分:加载,验证,准备,解析,初始化,下面我们就分别来看一下这五个过程. 加载 在加载阶段,虚拟机需要完成以下三件事情: 1)通过一个类的全限定名来获取定义此 ...
- 一夜搞懂 | JVM 类加载机制
前言 本文已经收录到我的Github个人博客,欢迎大佬们光临寒舍: 我的GIthub博客 学习导图 一.为什么要学习类加载机制? 今天想跟大家唠嗑唠嗑Java的类加载机制,这是Java的一个很重要的创 ...
随机推荐
- [转] 如何应用设计模式设计你的足球引擎(三和四)----Design Football Game(Part III and IV)
原文地址:http://www.codeproject.com/KB/cpp/applyingpatterns2.aspx 作者:An 'OOP' Madhusudanan 译者:赖勇浩(http:/ ...
- MVC登陆认证简单设置
首先,弄个基类 /// <summary> /// 所有控制器基类,里面重写了OnActionExecuted方法 /// </summary> public class Ba ...
- Mybatis中的jdbcType的作用
使用MyBatis框架做更新操作时,在该字段需要更新的内容为空时,就会出现1111错误,也就是无效的列类型,这个时候你就要使用jdbcType.至于什么时候要使用到javaType我还没遇到过,而且我 ...
- java:Filter、Listener 自定义拦截器和过滤器应用
一,Filter FilterEncoding 过滤器,统一设置servlet的编码格式. package com.dkt.filter; import java.io.IOException; im ...
- 14:求10000以内n的阶乘
14:求10000以内n的阶乘 查看 提交 统计 提问 总时间限制: 5000ms 内存限制: 655360kB 描述 求10000以内n的阶乘. 输入 只有一行输入,整数n(0<=n< ...
- 【转发】【小程序】微信小程序日常开发中常遇到的错误代码
还在为看不懂小程序错误状态码纠结吗?这里推荐一篇文章 重要的事情说三遍:原文链接 https://www.cnblogs.com/webonline/p/7528778.html 作者:玩世不恭. ...
- exim CVE-2017-16943 uaf漏洞分析
前言 本文由 本人 首发于 先知安全技术社区: https://xianzhi.aliyun.com/forum/user/5274 这是最近爆出来的 exim 的一个 uaf 漏洞,可以进行远程代码 ...
- Flex Box 简单弹性布局
弹性盒子模型有两种规范:早起的display:box 和后期的display:flex.它可以轻易的实现均分.浮动.居中等灵活布局,在移动端只考虑webkit内核时很实用. 一.display:box ...
- ubuntu16.04安装五笔输入法
在 Ubuntu16.04 中为fctix小企鹅增加五笔输入法 转载:https://jingyan.baidu.com/article/454316ab67d702f7a7c03a1a.html U ...
- springMVC入门-02
本节会在上节基础上讨论springMVC如何传值的问题. 在添加dispatcherServlet之后,拦截器会将url中的参数拦截下来,使之可以在controller中使用.以下代码就是在前台输入u ...