JVM知识总结(简单且高效)

1. JVM内存与本地内存

  • JVM内存:受虚拟机内存大小的参数控制,当大小超过参数设置的大小时会报OOM。
  • 本地内存:本地内存不受虚拟机内存参数的限制,只受物理内存容量的限制;虽然不受参数的限制,如果所占内存超过物理内存,仍然会报OOM。

2. JVM内存结构

  • 虚拟机栈:服务于Java方法
  • 本地方法栈:服务于本地方法
  • 程序计数器:保存当前线程执行的字节码位置,当然每个线程工作时都有独立的计数器。
  • 堆:用于存放对象
  • 方法区:用于存放常量、静态变量、数据类型、类信息等元数据
    在这里插入图片描述

3. 线程独占与共享区域

  • 独占:虚拟机栈、本地方法栈、程序计数器
  • 共享:堆、方法区

4. 栈与堆的区别

  • 栈:是运行时单位,代表逻辑,且区域连续
  • 堆:是存储单位,代表数据,区域不连续

5. 方法区、永久代、元空间

  • 方法区:是规范,是概念。
  • 永久代:是实现,在JDK7及之前版本,是方法区的一种实现。
  • 元空间:是实现,在JDK8及之后版本,是方法区的一种实现。(替代了永久代)
    在这里插入图片描述
    在这里插入图片描述

6. 元空间为什么替代了永久代

  • 突破内存限制,减少OOM。 由于元空间使用的是本地内存,而不是 JVM 内存。
  • 提高 Full GC 的效率。 因为永久代中存放了很多 JVM 需要的类信息,这些数据大多数是不会被清理的,所以 Full GC 往往无法回收多少空间。但在元空间模型中,由于字符串常量池已移至堆中,因此可以更有效地进行垃圾回收,避免了因频繁的 Full GC 导致的性能影响。
  • 满足不同的类加载需求和动态类加载的情况。 元空间可以动态地调整大小,在一些大型的、模块化的应用中,可能需要加载大量的类,这就需要大量的元数据存储空间。
  • 避免永久代调优和大小设置的复杂性。 在 Java8 之前的版本中,通常需要手动设置永久代的大小,以避免内存溢出的错误。这增加了应用的配置和管理的复杂性。而元空间使用本地内存,根据实际需求动态调整,大大简化了内存管理的复杂性。

7. JVM内存可见性

  • 线程对于变量的操作只能在自己的工作内存中进行,而不能直接对主存操作。
    在这里插入图片描述

8. 类的生命周期

  • 加载:通过类的完全限定名,查找此类字节码文件,并创建Class对象。
  • 验证:确保Class文件符合当前虚拟机的要求。
  • 准备:为static修饰的类变量分配内存,不包含final修饰的变量,因为final已经在编译时分配。
  • 解析:将常量池的符号引用替换为直接引用。
  • 初始化:静态块执行、静态变量赋值。
  • 使用:new出对象程序中使用。
  • 卸载:执行垃圾回收。
    在这里插入图片描述

9. 符号引用和直接引用

  • 符号引用即用**(用字符串符号的形式)**来表示引用,其实被引用的类、方法或者变量还没有被加载到内存中。而直接引用则是有具体引用地址的指针,被引用的类、方法或者变量已经被加载到内存中。

10. 类的初始化

只有对类主动使用时才会初始化,触发条件包括

  • 创建类的实例时。
  • 访问类的静态方法或静态变量的时候。
  • 使用Class.forName反射类的时候。
  • 某个子类初始化的时候。

11. 双亲委派加载机制

  • AppClassLoader从缓存查找类,没有则委托给父加载器ExtClassLoader
  • ExtClassLoader从缓存查找类,没有则委托给父加载器BootStrapClassLoader
  • BootStrapClassLoader从缓存查找类,没有则sun.mic.boot.class路径查找
  • BootStrapClassLoadersun.mic.boot.class路径查找,没有则让子类ExtClassLoader加载
  • ExtClassLoaderjava.ext.dirs路径查找,没有则让子类AppClassLoader加载
  • AppClassLoaderjava.class.path路径查找,如果找到就加载类,否则就抛出异常。
    在这里插入图片描述

12. 双亲委派机制的优点

  • 避免类的重复加载
  • 避免Java的核心API被篡改

13. GC如何判断对象可以被回收

  • 引用计数法:每个对象有一个引用计数属性,新增一个引用时计数加1,引用释放时计数减1,计数为0时可以回收
  • 可达性分析法:从GC Roots开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明此对象是不可用的,那么虚拟机就判断是可回收对象

14. 可达性分析算法

  • 可达性算法中的不可达对象并不是立即死亡的,对象拥有一次自我拯救的机会
  • 当对象编程(GC Roots)不可达时,GC会判断该对象是否执行过finalize方法,若执行了则直接回收。否则,若对象未执行过finalized方法,将其放入F-Queue队列。执行finalize方法完毕后,GC会再次判断该对象是否可达,若不可达,则进行回收,否则对象复活。

15. 四种JVM的垃圾回收算法

标记-清除算法:标记无用对象,然后进行清除回收。缺点:效率不高,无法清除垃圾碎片。
标记-整理算法:标记无用对象,让所有存活的对象都向一端移动,然后直接清除掉端边界以外的内存。
复制算法:按照容量划分二个大小相等的内存区域,当一块用完的时候将活着的对象复制到另一块上,然后再把已使用的内存空间一次清理掉。缺点:内存使用率不高,只有原来的一半。
分代算法:根据对象存活周期的不同将内存划分为几块,一般是新生代和老年代,新生代基本采用复制算法,老年代采用标记整理算法。

16. 分代算法

  1. new新对象放入堆中
  2. 如果在Eden区没有空间,那就发生Minor GC,保留存活的对象
  3. 当新生代仍然没有空间,那就将部分年龄大的新生代对象放入老年代
  4. 当老年代仍然没有空间,那就发生Major GC,保留存活的对象
  5. Major GC之后仍然没有空间,就会抛出OOM
    在这里插入图片描述

17. 为什么会出现OOM

  • 内存泄漏:申请使用完的内存没有释放,导致虚拟机不能再次使用该内存,此时这段内存就泄露了。因为申请者不用了,而又不能被虚拟机分配给别人用
  • 内存溢出:申请的内存超出了 JVM 能提供的内存大小,此时称之为溢出内存泄漏持续存在,最后一定会溢出,两者是因果关系

18. 常见的OOM报错

方法区溢出

  • 报错信息:java.lang.OutOfMemoryError: PermGen space
  • 常见问题:一般出现于大量Class对象,或者方法区过小

栈区溢出

  • 报错信息:java.lang.StackOverflowError
  • 常见问题:一般是由于程序中存在 死循环或者深度递归调用 造成的,或者栈区过小

堆区溢出

  • 报错信息:java.lang.OutOfMemoryError: Java heap space
  • 常见问题:内存泄漏、内存溢出

19. OOM如何解决

  • 主要使用dump文件jprofiler两个工具,具体自行学习

20. JVM的元空间中会发生垃圾回收吗

  • 垃圾回收不会发生在元空间,如果元空间满了或者是超过了临界值,会触发完全垃圾回收(FullGC)。
  • 如果正确设置元空间大小,那么就可以避免Full GC

21. 查看相关线程命令

  • jps(JVM Process Status Tool):显示指定系统内所有的HotSpot虚拟机进程。
  • jstat(JVM statistics Monitoring):显示出类装载、内存、垃圾收集、JIT编译等运行数据。
  • jmap(JVM Memory Map):命令用于生成heap dump文件
  • jhat(JVM Heap Analysis Tool):命令是与jmap搭配使用,用来分析jmap生成的dump,jhat内
    置了一个微型的HTTP/HTML服务器,生成dump的分析结果后,可以在浏览器中查看
  • jstack:用于生成java虚拟机当前时刻的线程快照。
  • jinfo(JVM Configuration info):这个命令作用是实时查看和调整虚拟机运行参数。

22. Minor GC与Full GC分别在什么时候发生?

  • 新生代内存不够用时候发生MGC也叫YGC,JVM内存不够的时候发生FGC。

23. 对象一定分配在堆中吗?有没有了解逃逸分析技术?

  • 不一定的,JVM通过「逃逸分析」,那些逃不出方法的对象会在栈上分配。

24. 什么是逃逸分析

  • 方法逃逸:在一个方法体内,定义一个局部变量,而它可能被外部方法引用,比如作为调用参数传递给方法,或作为对象直接返回。或者,可以理解成对象跳出了方法。
  • 线程逃逸:这个对象被其他线程访问到,比如赋值给了实例变量,并被其他线程访问到了。对象逃出了当前线程。
  • 逃逸分析:没有发生逃逸的对象,会进一步优化,将其放在栈中
// sb就逃逸了
public static StringBuffer craeteStringBuffer(String s1, String s2) {
    StringBuffer sb = new StringBuffer();
    sb.append(s1);
    sb.append(s2);
    return sb;
}

// sb没有逃逸了
public static String createStringBuffer(String s1, String s2) {
    StringBuffer sb = new StringBuffer();
    sb.append(s1);
    sb.append(s2);
    return sb.toString();
}

25. 逃逸分析带来的好处

  • 同步省略:如果对象只能一个线程被访问到,那么对于这个对象的操作可以不考虑同步。
  • 将堆分配转化为栈分配:对象可能是栈分配的候选,而不是堆分配。
  • 分离对象:有的对象可以,那么对象的部分(或全部)可以不存储在内存,而是存储在CPU寄存器中。

26. 逃逸分析的开启关闭

  • 从jdk 1.7开始已经默认开始逃逸分析,如需关闭,需要指定-XX:-DoEscapeAnalysis
  • -XX:+DoEscapeAnalysis : 表示开启逃逸分析
  • -XX:-DoEscapeAnalysis : 表示关闭逃逸分析

27. 同步省略
动态编译同步块的时候,JIT编译器可以借助逃逸分析来证实同步块的对象只能够被一个线程访问,就会取消对这部分代码的同步,这个过程也叫锁消除

如以下代码:

public void f() {
    Object hollis = new Object();
    synchronized(hollis) {
        System.out.println(hollis);
    }
}

hollis对象的生命周期只在f()方法中,并不会被其他线程所访问到,所以在JIT编译阶段优化成:

public void f() {
    Object hollis = new Object();
    System.out.println(hollis);
}

28. 元空间为什么能替代永久代

  • 区别:永久代存在于JVM限制内存,元空间存在于本地内存。所以元空间会更大。
  • 好处:能够避免方法区OOM异常,虽然我们可以通过设置永久代的大小来解决OOM,但是不是总能知道应该设置为多大合适, 如果使用默认值很容易遇到OOM错误。当使用元空间时,可以加载多少类的元数据就由系统的实际可用空间来控制啦,即元空间具有伸缩性。

29. Stop The World

  • 进行垃圾回收的过程中,会涉及对象的移动。为了保证对象引用更新的正确性,必须暂停所有的用户线程,像这样的停顿,虚拟机设计者形象描述为「Stop The World」,也简称STW。

30. 指针碰撞

  • 分配方式:一般情况下,JVM的对象都放在堆内存中(发生逃逸分析除外)。当类加载检查通过后,Java虚拟机开始为新生对象分配内存。如果Java堆中内存是绝对规整的,所有被使用过的的内存都被放到一边,空闲的内存放到另外一边,中间放着一个指针作为分界点的指示器,所分配内存仅仅是把那个指针向空闲空间方向挪动一段与对象大小相等的实例。
  • 指针碰撞:如果现在两个线程同时创建对象呢?
    在这里插入图片描述

31. 空闲列表

  • 如果Java堆内存中的内存并不是规整的,已被使用的内存和空闲的内存相互交错在一起,不可以进行指针碰撞啦,虚拟机必须维护一个列表,记录哪些内存是可用的,在分配的时候从列表找到一块大的空间分配给对象实例,并更新列表上的记录,这种分配方式就是空闲列表。

32. 什么是TLAB

  • 每个线程都有自己的一小块内存,这就是TLAB(Thread Local Allocation Buffer,本地线程分配缓存) 。虚拟机通过 -XX:UseTLAB 设定它的。

33. JVM有哪些垃圾回收器

  • Serial:新生代、单线程、复制算法(适用于小堆、单核)
  • ParScavenge:新生代、多线程、复制算法、高吞吐
  • Serial Old:老年代、单线程、标记-整理(适用于大堆、多核)
  • ParOld:老年代、多线程、标记-整理、低交互
  • CMS:老年代、多线程、标记-整理、低停顿低吞吐
  • G1:全堆、多线程、标记-整理

附录

「堆栈内存相关」

-Xms 设置初始堆的大小
-Xmx 设置最大堆的大小
-Xmn 设置年轻代大小,相当于同时配置-XX:NewSize和-XX:MaxNewSize为一样的值
-Xss 每个线程的堆栈大小
-XX:NewSize 设置年轻代大小(for 1.3/1.4)
-XX:MaxNewSize 年轻代最大值(for 1.3/1.4)
-XX:NewRatio 年轻代与年老代的比值(除去持久代)
-XX:SurvivorRatio Eden区与Survivor区的的比值
-XX:PretenureSizeThreshold 当创建的对象超过指定大小时,直接把对象分配在老年代。
-XX:MaxTenuringThreshold设定对象在Survivor复制的最大年龄阈值,超过阈值转移到
老年代

「垃圾收集器相关」

-XX:+UseParallelGC:选择垃圾收集器为并行收集器。
-XX:ParallelGCThreads=20:配置并行收集器的线程数
-XX:+UseConcMarkSweepGC:设置年老代为并发收集。
-XX:CMSFullGCsBeforeCompaction=5 由于并发收集器不对内存空间进行压缩、整理,
所以运行一段时间以后会产生“碎片”,使得运行效率降低。此值设置运行5次GC以后对内
存空间进行压缩、整理。
-XX:+UseCMSCompactAtFullCollection:打开对年老代的压缩。可能会影响性能,但是
可以消除碎片

「辅助信息相关」

-XX:+PrintGCDetails 打印GC详细信息
-XX:+HeapDumpOnOutOfMemoryError让JVM在发生内存溢出的时候自动生成内存快照,
排查问题用
-XX:+DisableExplicitGC禁止系统System.gc(),防止手动误触发FGC造成问题.
-XX:+PrintTLAB 查看TLAB空间的使用情况

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【Java集合篇】负载因子和容量的关系

负载因子和容量有什么关系 ✔️典型解析✔️loadfactor为啥默认是0.75F,不是1呢?✔️为什么HashMap的默认负载因子设置成0.75✔️0.75的数学依据是什么✔️0.75的必然因素 ✔️HashMap的初始值设为多少合适? ✔️典型解析 HashMap 中有几个属性,如 cap…

商智C店H5性能优化实战

前言 商智C店,是依托移动低码能力搭建的一个应用,产品面向B端商家。随着应用体量持续增大,考虑产品定位及用户体验,我们针对性能较差页面做了一次优化,并取得了不错的效果,用户体验值(UEI&…

每日一题——LeetCode1089.复写0

方法一 splice: 通过数组的slice方法,碰到 0就在后面加一个0,最后截取原数组的长度,舍弃后面部分。 但这样做是违反了题目的要求,不要在超过该数组长度的位置写入元素。 var duplicateZeros function(arr) {var le…

docker 完成MySQL的主从复制

文章目录 搭建步骤 搭建步骤 拉取镜像 docker pull mysql:5.7运行主从 docker run -p 3307:3306 --name mysql-master -v /mydata/mysql-master/log:/var/log/mysql -v /mydata/mysql-master/data:/var/lib/mysql -v /mydata/mysql-master/conf:/etc/mysql -e MYSQL_ROOT_P…

Springboot进行多环境配置的2种方式

本文来说下Springboot使用Spring Profile和Maven Profile进行多环境配置 文章目录 概述Spring Profile多环境主配置文件与不同环境的配置文件 Maven ProfileProfile配置资源过滤 Spring Profile与Maven Profile具体使用 概述 原因 在实际的项目上,一般会分三种环境d…

淘宝商品详情API接口(item_get-获得淘宝商品详情)主图,属性,sku,价格,搜索商品列表

淘宝开放平台提供了API接口,允许开发者获取淘宝商品的相关信息。为了获取商品详情,您可以使用 item_get API接口。以下是如何使用此API接口来获取商品的主图、属性、SKU、价格以及搜索商品列表的简要说明: 公共参数 名称类型必须描述keyStr…

如何利用MiniTab的命令行来提高数据建模效率

使用MiniTab进行数据建模时,如果涉及到需要多次更改数据、多次查看模型,感兴趣的同学可以尝试一下,把命令行显示出来,通过命令行的形式来执行,避免在繁多的菜单中到处查找。 操作方式如下图: 点击菜单“查…

Transformer架构和对照代码详解

1、英文架构图 下面图中展示了Transformer的英文架构,英文架构中的模块名称和具体代码一一对应,方便大家对照代码、理解和使用。 2、编码器 2.1 编码器介绍 从宏观⻆度来看,Transformer的编码器是由多个相同的层叠加⽽ 成的,每个…

Java重修第三天—“方法“的案例练习

案例一:买飞机票 题目 用户购买机票时,机票原价会按照淡季、旺季,头等舱还是经济舱的情况进行相应的优惠,优惠方案如下:5-10月为旺季,头等舱9折,经济舱8.5折。11月到来年4月为淡季,头等舱7折&…

内核线程创建-kthread_create

文章参考Linux内核线程kernel thread详解 - 知乎 大概意思就是早期创建内核线程,是交由内核处理,由内核自己完成(感觉好像也不太对呢),创建一个内核线程比较麻烦,会导致内核阻塞。因此就诞生了工作队列以及…

美格智能5G RedCap模组SRM813Q通过广东联通5G创新实验室测试认证

近日,美格智能5G RedCap轻量化模组SRM813Q正式通过广东联通5G创新实验室端到端的测试验收,获颁测评证书。美格智能已连续通过业内两家权威实验室的测试认证,充分验证SRM813Q系列模组已经具备了成熟的商用能力,将为智慧工业、安防监…

docker - 常用容器部署命令大全(MySQL、Redis、RabbitMQ、ES、Kibana、Nacos、Sentinel)

目录 一、常用容器运行指令 MySQL Redis RabbitMQ ElasticSearch & kibana Nacos Sentinel 一、常用容器运行指令 MySQL docker run -d --name mysql -p 3306:3306 -e TZAsia/Shanghai -e MYSQL_ROOT_PASSWORD1111 mysql:5.7 -e TZAsia/Shanghai:指定…

听GPT 讲Rust源代码--compiler(26)

File: rust/compiler/rustc_target/src/abi/call/mips.rs 在Rust源代码中的rust/compiler/rustc_target/src/abi/call/mips.rs文件是关于MIPS架构的函数调用ABI(Aplication Binary Interface)定义。ABI是编程语言与底层平台之间的接口规范,用于定义函数调用、参数传…

centos7部署minio单机版

一、目标 在centos7上部署minio单机版 二、centos7部署minio 1、下载minio mkdir /usr/local/minio cd /usr/local/minio wget https://dl.minio.io/server/minio/release/linux-amd64/minio chmod x minio 2、新建minio存储数据的目录 mkdir -p /data/minio/data3、新建…

首次引入大模型!Bert-vits2-Extra中文特化版40秒素材复刻巫师3叶奈法

Bert-vits2项目又更新了,更新了一个新的分支:中文特化,所谓中文特化,即针对中文音色的特殊优化版本,纯中文底模效果百尺竿头更进一步,同时首次引入了大模型,使用国产IDEA-CCNL/Erlangshen-Megat…

外包干了1个月,技术退步一大半。。。

先说一下自己的情况,本科生,19年通过校招进入广州某软件公司,干了接近4年的功能测试,今年年初,感觉自己不能够在这样下去了,长时间呆在一个舒适的环境会让一个人堕落!而我已经在一个企业干了四年的功能测试…

FFmpeg读取Assets资源文件

在Android开发中我们经常把原生资源文件放在assets目录下以供需要时读取,通过API提供的resources.assets.open(filename)/openFd(filenam)方法可以非常方便获得InputStream或FileDescriptor(文件标识符),但是在使用FFmpeg读取Asse…

Keil使用手册

文章目录 1 设置1.1 背景1.2 Project窗口显示.h文件1.3 注释1.4 Project窗口消失TAB转空格的设置keilsourceInsight 显示cannot evaluate普通局部变量静态全局变量静态局部变量 2 报错与解决2.1 warning:#1-D last line of file ends without anewline2.2 中文乱码 …

13. 强化学习编程实验1-在格子世界中寻宝(1)

文章目录 1.实验目的2.任务描述3.任务分析3.1 待求问题是多步决策问题否3.2 问题求解过程是一个马尔科夫决策过程3.3 状态空间S的确定3.4 动作空间A的确定3.5 状态转移概率P的确定3.6 立即回报R的确定3.7 折扣 γ \gamma γ的确定 4. 编程架构4.1 程序中有哪些对象和类4.2 环境…