【JVM性能优化】CMS回收器的Full-GC流程分析以及问题探究

简介: 【JVM性能优化】CMS回收器的Full-GC流程分析以及问题探究

CMS的7个步骤


image.png


重点步骤解读


初始标记(Initial Mark)

image.png

  • 目标:进行可达性分析,标记GC ROOT能直接关联到的对象
  • 标记范围:Young Gen + Old Gen
  • 线程:JDK1.7是单线程,JDK1.8是多线程
XX:+CMSParallelInitialMarkEnabled调整)
复制代码


  • STW:触发Stop-The-World
  • 特点:速度极快



并发标记(Concurrent Mark)

image.png


  • 目标:遍历阶段1初始标记出来的存活对象,然后继续递归标记这些对象可达的对象。(黑白灰三色标记法)
  • 标记范围:Young Gen + Old Gen
  • STW:不触发
  • 特点:慢,很耗时
  • 特殊操作:通过Old区卡片标记(Card Marking),提前把老年代空间逻辑划分为相等大小的区域(Card Page)。



如果引用关系发生改变,JVM会将发生改变的区域标记位“脏区”(Dirty Card)

JVM会对“并发标记”阶段应用线程新产生的对象及对象涉及的引用修改做记录(Mod-Union Table)




预清理(Preclean)


  • 目标:并发标记阶段中记录在(Mod-Union Table)的这些脏区会被找出来,刷新引用关系,清除“脏区”标记。
  • 标记范围: Old Gen
  • 线程:GC线程和应用线程也是并发执行
  • STW:不触发
  • 特点:速度一般




可中断的预清理(Concurrent Abortable Preclean)

image.png


  • 目标:降低垃圾回收时对应用的暂停时间,整个过程最耗时步骤在5(最终/重新标记),所以4步骤提前进行
  • 触发条件:此阶段在Eden区使用超过2M时启动,当然2M是默认的阈值,可以通过参数修改
  • 标记范围:Young Gen + Old Gen。


  • 处理From、To区对象,标记可达到老年代的对象。和3阶段一样,扫描处理Dirty Card中的对象。


  • 终止逻辑:CMS为了避免这个阶段没有等到Minor GC而陷入无限等待,提供了参数CMSMaxAbortablePrecleanTime ,默认为5s;含义是如果可中断的预清理执行超过5s,不管发没发生Minor GC,都会中止此阶段,进入Remark,以及eden去内存容量占比的空间阈值等


  • 线程:GC线程和应用线程也是并发执行。


  • STW:不触发
  • 特点:速度一般
  • 特殊:下一个阶段要执行最耗时且STW的Remark阶段,Remark阶段会将整个YoungGen作为GC-Root进行操作,如果4阶段能触发一次MonitorGC就再好不过了



重新标记(Final ReMark)


  • 目标:GC事件中第二次(也是最后一次)STW阶段,目的是完成老年代中所有存活对象的标记。
  • 标记范围:Young Gen + Old Gen。
  • 线程:GC线程独占执行
  • STW:触发Stop-The-World
  • 特点:速度较慢,可以说是整个CMS-Old GC的瓶颈点。



特殊:


  1. 遍历新生代对象,重新标记
  2. 根据GC Roots,重新标记
  3. 遍历老年代的Dirty Card,重新标记




并发清除(Concurrent Sweep)

image.png


  • 目标:根据标记结果清除垃圾对象。
  • 标记范围:Old Gen。
  • 线程:GC线程和应用线程也是并发执行
  • STW:不触发
  • 特点:速度一般。




并发重置(Concurrent Reset)


  • 目标:重置CMS算法相关的内部数据, 为下一次GC循环做准备。
  • 线程:GC线程和应用线程也是并发执行
  • STW:不触发
  • 特点:速度一般。





CMS相关常见问题总结


  1. CMS回收第一阶段“初始标记”都标记哪些?STW嘛?

初始标记需要STW,为了保障低停顿,只标记出GC-ROOT直接引用的对象。


  1. CMS回收第二阶段“并发标记”都标记哪些?STW嘛?

并发标记不会STW,垃圾回收线程与工作线程同时工作。并发标记时根据第一阶段标记的GC-ROOT进行延展标记。


  1. CMS回收第三阶段“重新标记”都标记哪些?STW嘛?

重新标记需要STW,需要标记第二阶段新创建对象&已有对象可能失去引用变成垃圾。


  1. CMS回收第四阶段“并发清理”会STW嘛?

并发清理很耗时,需要进行对象清理,垃圾回收线程与工作线程并发运行的,不会STW,可能会产生浮动对象产生1


  1. CMS回收“并发标记”&“并发清理”这两个阶段时,垃圾回收线程与工作线程并行运行,会导致cpu资源会成为瓶颈,CMS并发回收线程数如何设置?

cms默认启动的垃圾回收线程数:(cpu核数+3)/4

  1. CMS回收器在“并发清理”阶段可能会发生“Concurrent Mode Failure”问题?为什么?发生了“Concurrent Mode Failure”问题jvm会如何解决

CMS在“并发清理”阶段系统线程是工作着的,这就会产生“浮动垃圾”(通过YoungGC进入老年代的对象),CMS为了保证回收期间还有一定的内存空间让一些对象进入老年代,一半会预留一些空间,-XX:CMSInitiatingOccupancyFraction设置剩余百分比(默认68%),1.8以后变成了92%


  1. 如果CMS在回收期间,剩余空间小于本次晋升的对象大小会怎样呢?


就会发生“Concurrent Mode Failure”错误,也就是说并发垃圾回收失败,此时JVM会升级处理策略:自动启用“Serial Old”垃圾回收替代CMS回收,Serial Old 会强行STW,重新进行GC-Roots跟踪标记出全部垃圾对象,不允许新对象产生,一次性清理垃圾对象,然后恢复系统线程


  1. CMS内存碎片整理会STW嘛?为什么?


首先内存整理会STW,内存碎片会导致分配连续空间受阻,JVM就会频繁触发FullGC。所以CMS也会根据设置-XX:CMSFullGCsBeforeCompaction=n意思是说在上一次CMS并发GC执行过后,到底还要再执行多少次full GC才会做压缩

  • 默认是0,就是在默认配置下每次CMS GC顶不住了而要转入full GC的时候都会做压缩
  • 如果把CMSFullGCsBeforeCompaction配置为10,就会让上面说的第一个条件变成每隔10次真正的full GC才做一次压缩。
  1. 有几种情况会触发老年代执行GC?
  1. 老年代可用空间小于新生代全部对象的大小,如果没有开启空间担保策略会执行FullGC,一般默认担保策略是打开的
  2. 历次新生代GC后进入老年代的平均大小大于老年代可用空间。
  3. 老年代使用空间大于-XX:CMSInitiatingOccupancyFraction设置的阀值。
  1. 什么是分配担保?默认开启嘛?可以去掉分配担保机制吗?为什么需要的是连续空间?
  • 在发生Minor GC时候,虚拟机会检查老年代最大可用的连续空间是否大于新生代所有对象的总空间。


  • 如果大于,则此次Minor GC是安全的
  • 如果小于,则虚拟机会查看HandlePromotionFailure设置分配担保
  • HandlePromotionFailure=true,and 老年代最大可用连续空间是否大于历次晋升到老年代的对象的平均大小,如果大于,则尝试进行一次Minor GC


  • HandlePromotionFailure=false or 老年代最大可用连续空间小于历次晋升到老年代的对象的平均大小,执行FullGC


1.6以后是默认开启的。可以去掉,会加大FullGC发生的概率。虚拟机短暂停止,吞吐量、性能下降新生代使用的是复制算法,复制算法决定需要连续空间。
















相关文章
|
17天前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
38 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
6天前
|
存储 算法 Java
Java虚拟机(JVM)的内存管理与性能优化
本文深入探讨了Java虚拟机(JVM)的内存管理机制,包括堆、栈、方法区等关键区域的功能与作用。通过分析垃圾回收算法和调优策略,旨在帮助开发者理解如何有效提升Java应用的性能。文章采用通俗易懂的语言,结合具体实例,使读者能够轻松掌握复杂的内存管理概念,并应用于实际开发中。
|
17天前
|
小程序 Oracle Java
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
这篇文章是关于JVM基础知识的介绍,包括JVM的跨平台和跨语言特性、Class文件格式的详细解析,以及如何使用javap和jclasslib工具来分析Class文件。
27 0
JVM知识体系学习一:JVM了解基础、java编译后class文件的类结构详解,class分析工具 javap 和 jclasslib 的使用
|
17天前
|
算法 Java
JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
本文详细介绍了JVM中的GC算法,包括年轻代的复制算法和老年代的标记-整理算法。复制算法适用于年轻代,因其高效且能避免内存碎片;标记-整理算法则用于老年代,虽然效率较低,但能有效解决内存碎片问题。文章还解释了这两种算法的具体过程及其优缺点,并简要提及了其他GC算法。
 JVM进阶调优系列(4)年轻代和老年代采用什么GC算法回收?
|
24天前
|
存储 算法 Java
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
【JVM】垃圾释放方式:标记-清除、复制算法、标记-整理、分代回收
42 2
|
13天前
|
Java
JVM进阶调优系列(5)CMS回收器通俗演义一文讲透FullGC
本文介绍了JVM中CMS垃圾回收器对Full GC的优化,包括Stop the world的影响、Full GC触发条件、GC过程的四个阶段(初始标记、并发标记、重新标记、并发清理)及并发清理期间的Concurrent mode failure处理,并简述了GC roots的概念及其在GC中的作用。
|
18天前
|
算法 Java
JVM进阶调优系列(3)堆内存的对象什么时候被回收?
堆对象的生命周期是咋样的?什么时候被回收,回收前又如何流转?具体又是被如何回收?今天重点讲对象GC,看完这篇就全都明白了。
|
17天前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
29 4
|
3月前
|
Java Docker 索引
记录一次索引未建立、继而引发一系列的问题、包含索引创建失败、虚拟机中JVM虚拟机内存满的情况
这篇文章记录了作者在分布式微服务项目中遇到的一系列问题,起因是商品服务检索接口测试失败,原因是Elasticsearch索引未找到。文章详细描述了解决过程中遇到的几个关键问题:分词器的安装、Elasticsearch内存溢出的处理,以及最终成功创建`gulimall_product`索引的步骤。作者还分享了使用Postman测试接口的经历,并强调了问题解决过程中遇到的挑战和所花费的时间。
|
19天前
|
存储 缓存 算法
JVM核心知识点整理(内存模型),收藏再看!
JVM核心知识点整理(内存模型),收藏再看!
JVM核心知识点整理(内存模型),收藏再看!