《深入解析Java虚拟机:从JVM体系结构到垃圾回收算法》

文章目录

    • JVM体系结构
      • JVM的组成
    • 类加载器Class Loader
      • 类加载器的作用
        • 双亲委派机制
          • JVM自带三个类加载器
            • Bootstrap ClassLoader-根加载器
            • ExtClassLoader-扩展加载器
            • AppClassLoader-应用类加载器
    • Java历史-沙箱安全机制
      • 沙箱概念
      • 沙箱的作用
      • 本地代码和远程代码
      • 沙箱安全机制模型
        • JDK1 .0安全模型
        • JDK1 .1安全模型
        • JDK1 .2安全模型
        • 目前最新的安全模型
      • 沙箱安全机制的基本组件
        • 字节码校验器(bytecode verifier)
        • 类装载器(class loader)
        • 存取控制器(access controller)
        • 安全管理器(security manager)
        • 安全软件包(security package) :
    • Native与方法区
      • Native
      • 方法区
    • 深入理解栈
      • 关于main方法的调用
      • 栈运行的原理
        • 栈帧
    • HotSpot和堆
    • 新生区、老年区、永久区
      • 新生区(伊甸园区+幸存者区*2)
      • 老年区
      • 永久区
    • 堆内存调优
      • 报OOM怎么办?
        • 使用Jprofiler工具分析OOM原因
    • imgGC:垃圾回收
      • GC算法
        • 引用计数法
        • 复制算法
        • 标记清除算法
          • 标记清除
          • 标记清除压缩
      • 总结

来看一下常见的JVM面试题:

  • 请你谈谈对JVM的理解?java8虚拟机和之前的变化更新?
  • 什么是OOM,什么是栈溢出StackOverFlowError?怎么分析?
  • JVM的常用调优参数有哪些?
  • 内存快照如何抓取,怎么分析Dump文件?
  • 谈谈你对JVM中类加载器的认识?

如果回答不了上面的题目,那就开始接下来的学习JVM之路把~

JVM体系结构

Java程序是跑在我们的JVM虚拟机上的,而虚拟机又在操作系统之上。

下面的图要自己动手画一遍!

img

需要知道,操作系统是在硬件之上的。

img

栈这个区域是不会有垃圾的:

img

垃圾都在堆区,而方法区又是特殊堆:

img

jvm调优:99%都是在方法区和堆,大部分时间调堆。

img

JVM的组成

通过上图,我们已知JVM由三部分组成:

  • 类加载子系统
  • 运行时数据区
  • 执行引擎

类加载器Class Loader

类加载器的作用

类加载器的作用是把类(class)装在进内存里。

package com.linghu;/*** @author linghu* @date 2024/1/19 16:57*/
public class Car {public static void main(String[] args) {//类是模板,对象是具体的Car car1 = new Car();Car car2 = new Car();Car car3 = new Car();System.out.println("======对象=======");System.out.println(car1.hashCode());System.out.println(car2.hashCode());System.out.println(car3.hashCode());//这就是类的模板Class<? extends Car> aClass1 = car1.getClass();Class<? extends Car> aClass2 = car1.getClass();Class<? extends Car> aClass3 = car1.getClass();System.out.println("======类模板=======");System.out.println(aClass1.hashCode());System.out.println(aClass2.hashCode());System.out.println(aClass3.hashCode());}
}

分析如上代码和下图,可知:

类加载器Class Loader将类Car加载初始化成类模板。 也就是上面的aClass1aClass2aClass3 就是类模板,这三个都是同一个类模板

我们输出类模板的hash:

img

类模板的作用就是,类模板可以为我们的类实例化对象,也就是创建对象!我们上面的代码已经创建了三个对象:car1car2car3

我们输出三个对象的hash:

img

同样,我们也可以通过 类模板调用getClassLoader()得到类加载器,从而获取类的名字,这就是Java反射!

img

双亲委派机制

首先需要知道的是ClassLoader的作用就是将class文件加载到jvm虚拟机中去,JVM就可以正确运行了。但是,jvm启动的时候,并不会一次性加载所有的class文件,而是根据需要去动态加载。

  1. 类加载器在加载类的时候,优先委托给上级类加载器,其委托方向如为 AppClassLoader -> ExtClassLoader -> BootStrap

JVM自带的有三个类加载器:

JVM自带三个类加载器
Bootstrap ClassLoader-根加载器

最顶层的加载类。加载目录jre/lib/rt.jar里所有的class。

ExtClassLoader-扩展加载器

扩展的类加载器,加载目录%JRE_HOME%\lib\ext目录下的jar包和class文件。

AppClassLoader-应用类加载器

加载当前应用的classpath的所有类。 也就是我们自己定义的class。

  • 初始化顺序: Bootstrap ClassLoader->ExtClassLoader(加载路径:java.ext.dirs)->AppClassLoader(加载路径:java.class.path) 。
package com.linghu;/*** @author linghu* @date 2024/1/19 16:57*/
public class Car {public static void main(String[] args) {//类是模板,对象是具体的Car car1 = new Car();Car car2 = new Car();Car car3 = new Car();//        System.out.println("======对象=======");
//        System.out.println(car1.hashCode());
//        System.out.println(car2.hashCode());
//        System.out.println(car3.hashCode());//这就是类的模板Class<? extends Car> aClass1 = car1.getClass();
//        Class<? extends Car> aClass2 = car1.getClass();
//        Class<? extends Car> aClass3 = car1.getClass();ClassLoader classLoader = aClass1.getClassLoader();System.out.println(classLoader);//AppClassLoaderSystem.out.println(classLoader.getParent());//ExtClassLoader-->\jre\lib\extSystem.out.println(classLoader.getParent().getParent());//null,1、不存在}
}

img

img

总结双亲委派保证类加载器,自下而上的委派,又自上而下的加载,保证每一个类在各个类加载器中都是同一个类。一个非常明显的目的就是保证java官方的类库<JAVA_HOME>\lib和扩展类库<JAVA_HOME>\lib\ext的加载安全性,不会被开发者覆盖。

Java历史-沙箱安全机制

沙箱概念

Java安全模型的核心就是Java沙箱(sandbox)。沙箱机制就是将Java代码限定只能在虚JVM虚拟机中特定的运行范围,并且严格限制代码对本地系统资源访问,通过这样的方式来保证对Java代码的有效隔离,防止对本地操作系统造成破坏。

其实Windows也有这个沙箱的概念,像个内置虚拟机。

沙箱的作用

主要限制系统资源(CPU、内存、文件系统、网络)的访问。

不同级别的沙箱对系统资源访问的限制也有差异。

本地代码和远程代码

Java的执行程序分为:本地代码和远程代码。,

  • 本地代码:默认视为可信任的,可以访问一切本地资源。

  • 远程代码:被看作是不受信的。对于授信的本地代码,对于非授信的远程代码在早期的Java实现中,安全依赖于沙箱(Sandbox)机制。

沙箱安全机制模型

JDK1 .0安全模型

JDK1 .0安全模型本地代码可以访问系统资源,远程代码无法访问系统资源,比如用户希望远程代码访问本地系统的文件时候,就无法实现。

img

JDK1 .1安全模型

JDK1 .1 安全模型版本中,针对安全机制做了改进,增加了受信任安全策略,允许用户指定代码对本地资源的访问权限

img

JDK1 .2安全模型

JDK1 .2安全模型改进了安全机制,增加了代码签名。不论本地代码或是远程代码,统一按照用户的安全策略设定,由类加载器加载到虚拟机中权限不同的运行空间,从而来实现差异化的代码执行权限控制。

img

目前最新的安全模型

目前最新的安全模型引入了域 (Domain) 的概念。JVM虚拟机会把所有代码加载到不同的系统域和应用域,系统域部分专门负责与关键资源系统进行交互,而每个应用域部分则通过系统域的部分代理来对各种需要的资源进行精细划分然后可以进行访问。JVM虚拟机中不同的受保护域 (Protected Domain)对应不一样的权限 (Permission)。存在于不同域中的类文件就拥有了它所包含应用域所有可访问资源之和。

img

沙箱安全机制的基本组件

字节码校验器(bytecode verifier)

确保lava类文件遵循lava语言规范。这样可以帮助Java程序实现内存保护。但并不是所有的类文件都会经过字节码校验,比如核心类。

类装载器(class loader)
  • 防止恶意代码去干涉善意的代码,比如:双亲委派机制
  • 守护了被信任的类库边界;
  • 将代码归入保护域,确定了代码的权限范围可以进行哪些资源操作
存取控制器(access controller)

存取控制器可以控制核心API对操作系统的存取权限,用户可以设定控制策略。

安全管理器(security manager)

安全管理器主要是核心API和操作系统之间的主要接口。比如实现权限控制,比存取控制器优先级高。

安全软件包(security package) :

java.security下的类和扩展包下的类,允许用户为应用增加所需要安全特性:安全提供者、消息摘要、数字签名keytools、加密、鉴别。

Native与方法区

Native

package com.linghu;/*** @author linghu* @date 2024/1/23 9:53*/
public class Demo {public static void main(String[] args) {new Thread(()->{},"myThread").start();}private native void start0();
}

通过以上代码,可以看到有一个 start0方法。该方法用 native来进行修饰。有以下含义:

  • 凡事有 native修饰的,说明Java作用范围达不到了。回去调用C语言的库了~
  • 会进入本地方法栈
  • 调用本地方法接口JNI
private native void start0();

JNI作用:扩展Java的使用,融合不同的编程语言为Java所用!

img

Java诞生的时候,正是C和C++大行其道的时候,为了能够在市场上存活,Java在内存中单独开了一个标记区本地方法栈 Native Method Stack,登记Native方法,在最终执行的时候,加载本地方法库中的方法通过JNI。

img

Java程序驱动打印机、管理系统等等都需要用到这个Native。在企业级应用中较为少见!

方法区

Method Area 方法区。
方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊方法,如构造函数,接口代码也在此定义简单说,所有定义的方法的信息都保存在该区域,此区域属于共享区间;

静态变量、常量、类信息 (构造方法、接口定义)、运行时的常量池存在方法区中,但是 实例变量存在堆内存中,和方法区无关
static, final,Class,常量池

深入理解栈

栈:又叫栈内存。主管程序的运行和生命周期、线程同步。

线程结束,栈内存也就释放了。对于栈来说,不存在垃圾回收的说法、一旦线程结束,栈就Over!

栈内存中有:8大基本类型+对象引用+实例的方法。

关于main方法的调用

为什么main先执行,最后结束?

我们调用一个main方法,在main里调用test(),在test()调用a()方法,会出现内存溢出的问题,这个问题用栈表示:

img

test和a方法循环调用对方,最后会出现栈溢出的问题,也就是内存的问题,因为这个调用是没有限制的,无限循环!

img

栈运行的原理

栈帧

一个方法对应一个栈帧!其实这个可以理解成一个FCB!下面是两个栈帧!

栈帧(Stack Frame)是用于支持虚拟机进行方法调用和方法执行的数据结构。栈帧存储了方法的局部变量表、操作数栈、动态连接和方法返回地址等信息。每一个方法从调用至执行完成的过程,都对应着一个栈帧在虚拟机栈里从入栈到出栈的过程。

img

img

HotSpot和堆

我们通过 java -version命令查看:

img

我们用的虚拟机一般都是 HotSpot

Heap,一个JVM只有一个堆内存。堆内存的大小是可以调节的。

类加载器读取类文件后,一般会把什么东西放到堆中?

类、方法、常量、变量~保存所有引用类型的真实对象。

堆内存细分三个区域:

  • 新生区(伊甸园区)
  • 养老区
  • 永久区

堆也被称作GC堆

img

GC垃圾回收,主要是在伊甸园区和养老区~假设内存满了,报错OOM,堆内存不够!

img

JDK8以后,永久存储区改了个名字,叫元空间

新生区、老年区、永久区

其实通过上面的图就会发现:

新生区(伊甸园区+幸存者区*2)

  • 类诞生和死亡的地方
  • 伊甸园区:所有对象都是伊甸园区new出来的!
  • 幸存者区(0、1),轻GC定期清理伊甸园区,活下来的放入幸存者区,幸存者区满了以后重GC清理伊甸园区+幸存者区。活下来的放入养老区,都满了就报OOM!

真理:经过研究,99%的对象都是临时对象,直接被清理!

老年区

新生区剩下来的,轻GC杀不死的。

永久区

这个区域常驻内存,用来存放JDK自身携带的Class对象,interface元数据,存储的是java运行时一些环境和类信息,该区域不存在垃圾回收GC。关闭虚拟机就会自动释放这个内存。

  • jdk1.6之前:永久代,常量池在方法区
  • jdk1.7:永久代,但是慢慢退化了(去永久代),常量池在堆中
  • jdk1.8之后:无永久代,常量池在元空间

一个启动类,加载了大量第三方jar包,Tomcat部署了太多应用,大量动态生成的反射类。不断被加载。直到内存满,就会出现OOM。

img

方法区又称为非堆,本质还是堆,只是为了区分概念。

元空间逻辑上存在,物理上不存在。

img

堆内存调优

报OOM怎么办?

1、尝试扩大堆内存,如果还报错,说明有死循环代码或者垃圾代码。

2、分析内存,看一下哪个地方有问题(专业工具)

使用Jprofiler工具分析OOM原因

在一个项目中,突然出现了OOM的故障,该如何排除?研究为什么会出错?

  • 能够看到代码第几行出错:内存快照分析工具,MAT,Jprofiler
  • Debug,一行行分析代码!

MAT,Jprofiler作用:

  • 分析Dump内存文件,快速定位内存泄漏
  • 获得堆中的数据
  • 获得大的对象(大厂面试)

看如下代码:

package com.linghu;import java.util.ArrayList;/*** @author linghu* @date 2024/1/24 10:44*/
public class Demo03 {byte[] array = new byte[1*1024*1024]; //1mpublic static void main(String[] args) {ArrayList<Demo03> list = new ArrayList<>();int count = 0;try {while (true){list.add(new Demo03()); //不停地把创建对象放进列表,这是问题所在!count = count + 1;}} catch (Exception e) {System.out.println("count: "+count);e.printStackTrace();}}
}

报错如下

img

这个时候我们用Jprofiler工具分析OOM:

img

imgGC:垃圾回收

GC作用区域如下:

img

JVM在进行GC的时候,并不是对三个区域统一回收,大部分回收的是新生代。

  • 新生代
  • 幸存区
  • 老年区

GC分两种:

  • 轻GC
  • 重GC

关于GC面试题:

  • JVM的内存模型和分区~详细到每个分区放什么?
  • 堆里面的分区有哪些?Eden, from, to, 老年区,说说它们的特点!
  • GC算法有哪些?怎么用的?标记清除法,标记整理,复制算法,分代收集法。引用计数法。
  • 轻GC与重GC分别在什么时候发生?

GC算法

引用计数法

一般JVM不用,大型项目对象太多了!

img

引用次数为0的就会被清理掉!

img

复制算法

img

img

  • 好处:没有内存的碎片,内存效率高。
  • 坏处:浪费了内存空间(一个幸存空间永远是空的);假设对象100%存活,复制成本很高。

复制算法最佳使用场景:对象存活度较低的时候,新生区。

标记清除算法
标记清除

img

  • 优点:不需要额外空间,优化了复制算法
  • 缺点:两次扫描,严重浪费时间,会产生内存碎片
标记清除压缩

三部曲:

  • 标记
  • 清除
  • 压缩

img

总结

  • 内存效率:复制算法>标记清除算法>标记压缩算法(时间复杂度)
  • 内存整齐度:复制算法=标记压缩算法>标记清除算法
  • 内存利用率:标记压缩算法=标记清除算法>复制算法

难道没有最优算法吗?

答案:无,没有最好的算法,只有合适的算法(GC也被称为分代收集算法)。

  • 年轻代:存活率低,用复制算法。

  • 老年代:存活率高,区域大,用标记-清除-压缩。

    参考和研究:《深入理解Java虚拟机》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/644184.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言快速排序(非递归)图文详解

前言&#xff1a; 上一期分析了快速排序的三种写法&#xff0c;这三种写法有一个相同点&#xff0c;都是采用递归形式来实现的&#xff0c;那么有没有非递归的方法实现呢&#xff1f;答案是当然有&#xff0c;用非递归的方法实现快速排序&#xff0c;其实可以借助数据结构中的栈…

【LangChain学习之旅】—(10) 用RouterChain确定客户意图

【【LangChain学习之旅】—&#xff08;10&#xff09; 用RouterChain确定客户意图 任务设定整体框架具体步骤如下&#xff1a; 具体实现构建提示信息的模板构建目标链 Reference&#xff1a;LangChain 实战课 任务设定 首先&#xff0c;还是先看一下今天要完成一个什么样的任…

《鸟哥的Linux私房菜》第1章——总结与简答题回答

目录 一、什么是Linux&#xff0c;有什么作用&#xff1f; 二、Linux发展史 三、Linux发行版 四、简答题部分 一、什么是Linux&#xff0c;有什么作用&#xff1f; Linux包括内核和系统调用两部分&#xff0c;是位于硬件设备与应用程序中间的操作系统。 操作系统其实也是…

多维时序 | Matlab实现GWO-TCN-Multihead-Attention灰狼算法优化时间卷积网络结合多头注意力机制多变量时间序列预测

多维时序 | Matlab实现GWO-TCN-Multihead-Attention灰狼算法优化时间卷积网络结合多头注意力机制多变量时间序列预测 目录 多维时序 | Matlab实现GWO-TCN-Multihead-Attention灰狼算法优化时间卷积网络结合多头注意力机制多变量时间序列预测效果一览基本介绍程序设计参考资料 效…

python爬虫代码示例:爬取京东详情页图片

python爬虫代码示例:爬取京东详情页图片 一、Requests安装及示例 爬虫爬取网页内容首先要获取网页的内容&#xff0c;通过requests库进行获取。 GitHub: https://github.com/requests/requests PyPl: https://pypi.python.org/pypi/requests 官方文档:http://wwwpython-requ…

Java可以用于物联网的开发吗?

Java可以用于物联网的开发吗? 在开始前我有一些资料&#xff0c;是我根据网友给的问题精心整理了一份「Java的资料从专业入门到高级教程」&#xff0c; 点个关注在评论区回复“888”之后私信回复“888”&#xff0c;全部无偿共享给大家&#xff01;&#xff01;&#xff01;J…

web安全学习笔记【06】——http\https抓包

思维导图放最后 #知识点&#xff1a; 1、Web常规-系统&中间件&数据库&源码等 2、Web其他-前后端&软件&Docker&分配站等 3、Web拓展-CDN&WAF&OSS&反向&负载均衡等 ----------------------------------- 1、APP架构-封装&原生态&…

SpringBoot集成mybatis时idea控制台中文乱码问题解决

在application.yml中配置好映射文件打印数据库日志文件时&#xff0c;控制台出现乱码的情况解决如下 问题 在执行查询操作的时候&#xff0c;查询时可以查看是没有问题的&#xff0c;但是控制台乱码了 解决 在File-Setting-Editor-File Encodings中设置如图所示就可以了 现在…

软件测试之功能测试详解

一、测试项目启动与研读需求文档 &#xff08;一&#xff09; 组建测试团队 1、测试团队中的角色 2、测试团队的基本责任 尽早地发现软件程序、系统或产品中所有的问题。督促和协助开发人员尽快地解决程序中的缺陷。帮助项目管理人员制定合理的开发和测试计划。对缺陷进行跟…

业余爱好-生物信息学/生物化学/物理/统计学/政治/数学/概率论/AI/AGI/区块链

生物信息学 高等数学—元素和极限-实数的定义高等数学—元素和极限-实数的元素个数高等数学—元素和极限-自然数个数少于实数个数高等数学—元素和极限-无穷大之比较高等数学—元素和极限-级数的收敛高等数学—元素和极限-极限的定义数学分析与概率论人工智能AI数学基础——全套…

http503错误是什么原因

HTTP503错误在站长圈很经常遇到&#xff0c;很多网站站长经常遇到的HTTP503错误经常会不知道怎么去解决它。今天我们就来针对HTTP503错误问题展开说说。HTTP503错误是指服务器暂时无法处理客户端的请求&#xff0c;常常出现在服务器超负荷或维护期间。在这种情况下&#xff0c;…

HTML 入门手册(二)

目录 HTML 入门手册(一) 10-表单 11-input标签 11.1文本框 (text) 11.2密码框 (password) 11.3单选按钮 (radio) 11.4复选框 (checkbox) 11.5普通按钮 11.6提交按钮 (submit) 11.7重置按钮 (reset) 11.8隐藏域 (hidden) 11.9文件上传 (file) 11.10数字输入 (numbe…

2024年上海高考数学最后四个多月的备考攻略,目标140+

亲爱的同学们&#xff0c;寒假已经来临&#xff0c;春节即将到来&#xff0c;距离2024年上海高考已经余额不足5个月了。作为让许多学子头疼&#xff0c;也是拉分大户的数学科目&#xff0c;你准备好了吗&#xff1f;今天&#xff0c;六分成长为您分享上海高考数学最后四个多月的…

Sulfo Cy3 hydrazide,磺化-Cy3-酰肼,可用于与生物分子的羰基衍生物偶联

您好&#xff0c;欢迎来到新研之家 文章关键词&#xff1a;Sulfo-Cyanine3-hydrazide&#xff0c;Sulfo Cy3 hydrazide&#xff0c;Sulfo Cyanine3 HZ&#xff0c;磺化 Cy3 酰肼&#xff0c;磺化-Cy3-酰肼 一、基本信息 产品简介&#xff1a;Sulfo-Cyanine3-hydrazide能够与…

【GitHub项目推荐--人脸识别】【转载】

01 带有移动应用程序的人脸识别库 OpenFace 作为用于人脸识别的通用库&#xff0c;能够实现瞬态和移动人脸识别&#xff0c;目前在 GitHub 上斩获 14291 Star。以下为 LFW 数据集 Sylvestor Stallone 输入单个图像的流程。 项目地址&#xff1a;https://github.com/cmusatya…

用python写一个双色球中奖模拟器

先说规则&#xff1a; 一等奖&#xff1a;投注号码与当期开奖号码全部相同&#xff08;顺序不限&#xff0c;下同&#xff09;&#xff0c;即中奖&#xff1b; 二等奖&#xff1a;投注号码与当期开奖号码中的6个红色球号码相同&#xff0c;即中奖&#xff1b; 三等奖&#xf…

ai智能写作软件有分享吗?分享4款解放双手的软件!

随着人工智能技术的不断发展&#xff0c;AI智能写作软件逐渐成为内容创作者们的新宠。这些软件不仅能够帮助我们快速生成高质量的文本内容&#xff0c;还能在优化搜索引擎排名等方面发挥重要作用。本文将为大家介绍几款常用的AI智能写作软件&#xff0c;让您轻松提升内容创作效…

科技发展趋势,墨水屏电子桌牌将发挥更重要作用

随着科技的不断发展&#xff0c;电子桌牌作为信息展示和宣传的新型设备&#xff0c;逐渐在各个行业得到广泛应用。在国企单位、政府部门、大企业、外企等&#xff0c;墨水屏电子桌牌作为一种新型的数字化展示工具&#xff0c;也已经得到了越来越多的应用。下面&#xff0c;中科…

2023年智能可穿戴行业市场分析(电商数据查询分析):智能手表销额增长21%,手环明显下滑

近年来&#xff0c;随着技术的进步&#xff0c;智能可穿戴设备在社交网络、医疗保健、导航等诸多领域有着非常广泛的应用&#xff0c;这为大众生活带来了诸多便利。 当前的可穿戴产品形态纷繁多样&#xff0c;主要包括智能手表、智能眼镜、智能手环、健康穿戴和体感控制等等&am…

“接口”公共规范的遵守者

&#x1f468;‍&#x1f4bb;作者简介&#xff1a;&#x1f468;&#x1f3fb;‍&#x1f393;告别&#xff0c;今天 &#x1f4d4;高质量专栏 &#xff1a;☕java趣味之旅 欢迎&#x1f64f;点赞&#x1f5e3;️评论&#x1f4e5;收藏&#x1f493;关注 &#x1f496;衷心的希…