讲透JVM类加载机制,向高手进阶!

2023-11-08

目录:

  1. 前言
  2. JVM在什么情况下会加载一个类?
  3. 从实用角度出发,来看看验证、准备和初始化的过程
  4. 核心阶段:初始化
  5. 类加载器和双亲委派机制

1、前言

先来看一下JVM整体的一个运行原理。

我们首先从“.java”代码文件,编译成“.class”字节码文件,然后类加载器把“.class”字节码文件中的类给加载到JVM中,接着是JVM来执行我们写好的那些类中的代码,整体是这么个顺序。

我们看看下图,感受一下这个过程:

在这里插入图片描述

那么今天,我们就来仔细看看上图中的“类加载”这个过程,看看JVM的类加载机制到底是怎么样的?

搞清楚这个过程了,那么以后在面试时,对面试官常问的JVM类加载机制,就能把一些核心概念说清楚了。


2、JVM在什么情况下会加载一个类?

其实类加载过程非常的琐碎复杂,但是对于我们平时从工作中实用的角度来说,主要是把握他的核心工作原理就可以。

一个类从加载到使用,一般会经历下面的这个过程:

加载 -> 验证 -> 准备 -> 解析 -> 初始化 -> 使用 -> 卸载

所以首先要搞明白的第一个问题,就是JVM在执行我们写好的代码的过程中,一般在什么情况下会去加载一个类呢?

也就是说,啥时候会从“.class”字节码文件中加载这个类到JVM内存里来。

其实答案非常简单,就是在你的代码中用到这个类的时候

举个简单的例子,比如下面你有一个类(Kafka.class),里面有一个“main()”方法作为主入口。

那么一旦你的JVM进程启动之后,它一定会先把你的这个类(Kafka.cass)加载到内存里,然后从“main()”方法的入口代码开始执行。

在这里插入图片描述

我们还是坚持一步一图,大家先看看下图,感受一下:

在这里插入图片描述

接着假设上面的代码中,出现了如下的这么一行代码:

在这里插入图片描述

这时可能大家就想了,你的代码中明显需要使用“ReplicaManager”这个类去实例化一个对象,此时必须得把“ReplicaManager.class”字节码文件中的这个类加载到内存里来啊!是不是

所以这个时候就会触发JVM通过类加载器,从“ReplicaManager.class”字节码文件中加载对应的类到内存里来使用,这样代码才能跑起来。

我们来看下面的图:

在这里插入图片描述

上面就是给大家举的一个例子,相信非常的通俗易懂。

简单概括一下:首先你的代码中包含“main()”方法的主类一定会在JVM进程启动之后被加载到内存,开始执行你的“main()”方法中的代码

接着遇到你使用了别的类,比如“ReplicaManager”,此时就会从对应的“.class”字节码文件加载对应的类到内存里来。


3、从实用角度出发,来看看验证、准备和初始化的过程

其实上面的类加载时机的问题,对于很多有经验的同学来说不是什么问题。但是对于很多初学者来说,是一个非常重要的需要捋清的概念。

接下来就来简单带着大家,从实用的角度出发,过一下另外三个概念:

验证、准备、初始化

其实对于这三个概念,没太大的必要去深究里面的细节,这里的细节很多很繁琐,对于大部分同学而言,只要脑子里有下面的几个概念就可以了:

(1)验证阶段

简单来说,这一步就是根据Java虚拟机规范,来校验你加载进来的“.class”文件中的内容,是否符合指定的规范。

这个相信很好理解,假如说,你的“.class”文件被人篡改了,里面的字节码压根儿不符合规范,那么JVM是没法去执行这个字节码的!

所以把“.class”加载到内存里之后,必须先验证一下,校验他必须完全符合JVM规范,后续才能交给JVM来运行。

下面用一张图,展示了这个过程:

在这里插入图片描述

(2)准备阶段

这个阶段其实也很好理解,咱们都知道,我们写好的那些类,其实都有一些类变量,比如下面的这个“ReplicaManager”类:

在这里插入图片描述

假设你有这么一个“ReplicaManager”类,他的“ReplicaManager.class”文件内容刚刚被加载到内存之后,会进行验证,确认这个字节码文件的内容是规范的。

接着,就会进行准备工作,这个准备工作,其实就是给这个“ReplicaManager”类分配一定的内存空间。

然后给他里面的类变量(也就是static修饰的变量)分配内存空间,来一个默认的初始值。

比如上面的示例里,就会给“flushInterval”这个类变量分配内容空间,给一个“0”这个初始值。

整个过程,如下图所示:

在这里插入图片描述

(3)解析阶段

这个阶段干的事儿,实际上是把符号引用替换为直接引用的过程,其实这个部分的内容很复杂,涉及到JVM的底层

但是注意,同学们,就我本意而言,希望第一周的文章,绝对是浅显易懂的,循序渐进,要保证每个同学都能绝对看懂。

所以针对这个阶段,现在不打算做过深的解读,因为从实用角度而言,对很多同学在工作中实践JVM技术其实也用不到,所以这里大家就暂时知道有这么一个阶段就可以了。

同样,我还是给大家画图展示一下:

在这里插入图片描述

(4)三个阶段的小结

其实这三个阶段里,最核心的大家务必关注的,就是“准备阶段

因为这个阶段是给加载进来的类分配好了内存空间,类变量也分配好了内存空间,并且给了默认的初始值,这个概念,大家心里一定要有。


4、核心阶段:初始化

之前说过,在准备阶段时,就会把我们的“ReplicaManager”类给分配好内存空间

另外他的一个类变量“flushInterval”也会给一个默认的初始值“0”,那么接下来,在初始化阶段,就会正式执行我们的类初始化的代码了。

那么什么是类初始化的代码呢?我们来看看下面这段代码:

在这里插入图片描述

大家可以看到,对于“flushInterval”这个类变量,我们是打算通过Configuration.getInt(“replica.flush.interval”)这段代码来获取一个值,并且赋值给他的

但是在准备阶段会执行这个赋值逻辑吗

NO!在准备阶段,仅仅是给“flushInterval”类变量开辟一个内存空间,然后给个初始值“0”罢了。

那么这段赋值的代码什么时候执行呢?答案是在“初始化”阶段来执行。

在这个阶段,就会执行类的初始化代码,比如上面的 Configuration.getInt(“replica.flush.interval”) 代码就会在这里执行,完成一个配置项的读取,然后赋值给这个类变量“flushInterval”。

另外比如下图的static静态代码块,也会在这个阶段来执行。

类似下面的代码语义,可以理解为类初始化的时候,调用“loadReplicaFromDish()”方法从磁盘中加载数据副本,并且放在静态变量“replicas”中:

在这里插入图片描述

那么搞明白了类的初始化是什么,就得来看看类的初始化的规则了。

什么时候会初始化一个类?

一般来说有以下一些时机:比如“new ReplicaManager()”来实例化类的对象了,此时就会触发类的加载到初始化的全过程,把这个类准备好,然后再实例化一个对象出来;

或者是包含“main()”方法的主类,必须是立马初始化的。

此外,这里还有一个非常重要的规则,就是如果初始化一个类的时候,发现他的父类还没初始化,那么必须先初始化他的父类

比如下面的代码:

在这里插入图片描述

如果你要“new ReplicaManager()”初始化这个类的实例,那么会加载这个类,然后初始化这个类

但是初始化这个类之前,发现AbstractDataManager作为父类还没加载和初始化,那么必须先加载这个父类,并且初始化这个父类。

这个规则,大家必须得牢记,再来一张图,借助图片来进行理解:

在这里插入图片描述

5、类加载器和双亲委派机制

现在相信大家都搞明白了整个类加载从触发时机到初始化的过程了,接着给大家说一下类加载器的概念。因为实现上述过程,那必须是依靠类加载器来实现的。

那么Java里有哪些类加载器呢?简单来说有下面几种:

(1)启动类加载器

Bootstrap ClassLoader,他主要是负责加载我们在机器上安装的Java目录下的核心类的

相信大家都知道,如果你要在一个机器上运行自己写好的Java系统,无论是windows笔记本,还是linux服务器,是不是都得装一下JDK?

那么在你的Java安装目录下,就有一个“lib”目录,大家可以自己去找找看,这里就有Java最核心的一些类库,支撑你的Java系统的运行。

所以一旦你的JVM启动,那么首先就会依托启动类加载器,去加载你的Java安装目录下的“lib”目录中的核心类库。

(2)扩展类加载器

Extension ClassLoader,这个类加载器其实也是类似的,就是你的Java安装目录下,有一个“lib\ext”目录

这里面有一些类,就是需要使用这个类加载器来加载的,支撑你的系统的运行。

那么你的JVM一旦启动,是不是也得从Java安装目录下,加载这个“lib\ext”目录中的类?

(3)应用程序类加载器

Application ClassLoader,这类加载器就负责去加载“ClassPath”环境变量所指定的路径中的类

其实你大致就理解为去加载你写好的Java代码吧,这个类加载器就负责加载你写好的那些类到内存里。

(4)自定义类加载器

除了上面那几种之外,还可以自定义类加载器,去根据你自己的需求加载你的类。

(5)双亲委派机制

JVM的类加载器是有亲子层级结构的,就是说启动类加载器是最上层的,扩展类加载器在第二层,第三层是应用程序类加载器,最后一层是自定义类加载器。

大家看下图:

在这里插入图片描述

然后,基于这个亲子层级结构,就有一个双亲委派的机制

什么意思呢?

就是假设你的应用程序类加载器需要加载一个类,他首先会委派给自己的父类加载器去加载,最终传导到顶层的类加载器去加载

但是如果父类加载器在自己负责加载的范围内,没找到这个类,那么就会下推加载权利给自己的子类加载器。

听完了上面一大堆绕口令,是不是很迷茫?别着急,咱们用一个例子来说明一下。

比如你的JVM现在需要加载“ReplicaManager”类,此时应用程序类加载器会问问自己的爸爸,也就是扩展类加载器,你能加载到这个类吗?

然后扩展类加载器直接问自己的爸爸,启动类加载器,你能加载到这个类吗?

启动类加载器心想,我在Java安装目录下,没找到这个类啊,自己找去

然后,就下推加载权利给扩展类加载器这个儿子,结果扩展类加载器找了半天,也没找到自己负责的目录中有这个类。

这时他很生气,说:明明就是你应用程序加载器自己负责的,你自己找去。

然后应用程序类加载器在自己负责的范围内,比如就是你写好的那个系统打包成的jar包吧,一下子发现,就在这里!然后就自己把这个类加载到内存里去了。

这就是所谓的双亲委派模型:先找父亲去加载,不行的话再由儿子来加载。

这样的话,可以避免多层级的加载器结构重复加载某些类。

最后,给大家来一张图,感受一下类加载器的双亲委派模型。

在这里插入图片描述

V-xin:ruyuanhadeng获得600+页原创精品文章汇总PDF

另外推荐儒猿课堂的1元系列课程给您,欢迎加入一起学习~

互联网Java工程师面试突击课(1元专享)

SpringCloudAlibaba零基础入门到项目实战(1元专享)

亿级流量下的电商详情页系统实战项目(1元专享)

Kafka消息中间件内核源码精讲(1元专享)

12个实战案例带你玩转Java并发编程(1元专享)

Elasticsearch零基础入门到精通(1元专享)

基于Java手写分布式中间件系统实战(1元专享)

基于ShardingSphere的分库分表实战课(1元专享)

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

讲透JVM类加载机制,向高手进阶! 的相关文章

随机推荐

  • python安装程序打不开_使用PIP安装Python包会导致链接:致命错误LNK1104:无法打开文件“python27.lib”...

    我试图让PIP安装Twisted 但我得到了这个链接错误 我使用的是64位Windows8和Python2 7 10 我认为这个问题与visualstudio编译器有关 我有2009年和2010年 但我不确定它用的是哪一个 我已经看过大量关
  • 分布式的环境下, MySQL和Redis如何保持数据的一致性?

    问题 一台MySQL 一台Redis 两台应用服务器 用户的数据存储持久化在MySQL中 缓存在Redis 有请求的时候从Redis中获取缓存的用户数据 有修改则同时修改MySQL和Redis中的数据 现在问题是 1 先保存到MySQL和先
  • 安装python遇到错误_安装python2.7出错怎么解决

    Python是一种跨平台的计算机程序设计语言 是一种面向对象的动态类型语言 最初被设计用于编写自动化脚本 shell 随着版本的不断 更新和语言新功能的添加 越来越多被用于独立的 大型项目的开发 首先登录python官网 选择要下载的安装包
  • ElementUI浅尝辄止23:Loading 加载

    Loading加载组件 加载数据时显示动效 常见于加载数据量大的业务操作 附带动态效果 1 如何使用 区域加载 在表格等容器中加载数据时显示 Element 提供了两种调用 Loading 的方法 指令和服务 对于自定义指令v loadin
  • linux重置root用户密码

    重置root密码 法一 rd break 第 1 步 重启系统编辑内核参数 第 2 步 找到 linux 这行 在此行末尾空格后输入rd break End键也可直接进入行尾 成功后显示页面为 第 3 步 查看 可选 该步骤可省略 命令为
  • 把一个数组的第三到第六位之间的元素删除 形成一个新的数组

    思路一 找到第三和第六之间的4和5位 也就是下标位3和4的元素 删除之后再前移 思路二 找到第三和第六之间的4和5位 也就是下标位3和4的元素 运用if中continue关键字跳过这两个位置 思路一和思路二的区别是思路一元素前移最后两位是0
  • fio测试磁盘性能

    rwmixwrite是fio命令中一个用于指定读写比例的参数 它的含义是在随机读写的情况下 写操作的百分比 例如 rwmixwrite 30表示30 的操作是写操作 70 的操作是读操作 以下是一个示例命令 fio name mytest
  • 帝国cms发送ajax请求,[分享]美化ajax弹出的提示信息

    前台页面只需要在内容模板里面加上这两个文件 然后再修改一下 e data js ajax js var http request false function makeRequest url functionName httpType
  • 收藏学习!15个使用率超高的Python库

    今天给大家分享最近一年内PyPI上下载量最高的Python包 现在我们来看看这些包的作用 他们之间的关系 以及为什么如此流行 1 Urllib3 8 93亿次下载 Urllib3 是 Python 的 HTTP 客户端 它提供了许多 Pyt
  • Keil 5出现Error: L6218E: Undefined symbol解决方法

    目录 1 找到相关未定义函数 选中然后右击 GO TO Reference To ADC Cmd 2 发现该函数在stm32f10x adc h中 3 可以看到右边fwlib里没有相应 c文件 我们添加上 4 右击 选择manage pro
  • HTML怎么插入一段代码

    之前学习了点HTML的基础知识 在工作中给别人发邮件的时候 使用HTML插入超链接 这回需要插入一段代码块 语法 pre pre 将需要传入的代码放在 pre 和 pre 之间就可以了 浏览器自动识别 示例 p 这是一段代码块 p pre
  • linux查看内存占用情况(top、free、ps)

    top命令 top命令是Linux下常用的性能分析工具 能够实时显示系统中各个进程的资源占用状况 类似于Windows的任务管理器 运行 top 命令后 CPU 使用状态会以全屏的方式显示 并且会处在对话的模式 用基于 top 的命令 可以
  • Embedded Linux Conference Europe schedule posted

    From http linuxgizmos com embedded linux conference europe schedule posted The Linux Foundation posted a schedule for Li
  • 2023 年程序员的最佳工作角色

    当今价值数十亿美元的计算机编程市场正受到下一代应用程序和产品的日益普及的推动 例如自动编码 统计计算 数据分析 ML 机器学习 和 AI 人工智能 根据MarketsandMarkets的一项研究 到11年 全球CAC 计算机辅助编码 行业
  • Redis三种集群架构

    一 主从架构 搭建主从结构 从节点配置步骤 1 复制一份redis conf文件 2 将相关配置修改为如下值 port 6380 pidfile var run redis 6380 pid 把pid进程号写入pidfile配置的文件 lo
  • 统计学第九周:参数估计python实现

    统计学第九周 参数估计复习 参数估计 根据从总体中随机取样获得样本 根据取样样本来估计总体分布中参数的过程 方法 估计形式上分 点估计与区间估计 估计的方法有矩法估计 最小二乘法估计 似然估计 贝叶斯估计等等 问题一般有 未知参数的估计量
  • 期货方法(期货方法很简单 只用MACD)

    期货法 在期货市场上 95 的人都是赔钱的 对于期货市场的新手来说 一定要有一个清晰的认识 95 的数字告诉我们什么 它只是告诉我们 在获得稳定的利润之前 我们不应该投入太多的钱 对于初学者来说 投入的钱越多 损失的钱就越多 所以一定要控制
  • 渗透之Aircrack—wifi破解

    环境 kali 工具使用 无线网卡 型号雷凌RT3070L 注意型号选择的使用 不同型号可能导致kali无法识别 步骤总结 1 kali连接外接无线网卡 2 ifconfig查看网卡信息 新出现的wlan0网卡 3 开启网卡监听模式 air
  • 实践积累:用Vue3简单写一个单行横向滚动组件

    目录 效果图 需求分析 实现分析 样式展示分析 变量分析 方法分析 实现步骤 1 实现模板 2 实现css 3 首先获取list 4 页面挂载后监听groupBoxRef的scroll事件并获取当前的滚动位置 5 计算展示的宽度显隐箭头 当
  • 讲透JVM类加载机制,向高手进阶!

    目录 前言 JVM在什么情况下会加载一个类 从实用角度出发 来看看验证 准备和初始化的过程 核心阶段 初始化 类加载器和双亲委派机制 1 前言 先来看一下JVM整体的一个运行原理 我们首先从 java 代码文件 编译成 class 字节码文