jvm - CMS收集器

终结者T800
• 阅读 1460

CMS是基于标记-清除算法的,收集的时候分为4个步骤:

  1. 初始标记
  2. 并发标记
  3. 重新标记
  4. 并发清除

# 初始标记
初始标记仅仅只是标记一下GC Roots能直接关联到的对象,所以速度很快。比如下图,这边的GC Roots只用了虚拟机栈为例。两个虚拟机栈分表创建了对象OBJ_A1和OBJ_B1,他们也各有自己的其他引用,在这个阶段,他只会标记OBJ_A1和OBJ_B1,其他的引用是不标记的,所以尽管这个阶段有STW,但是标记的数量少,时间很快,基本不影响。
jvm - CMS收集器

并发标记

并发标记就是根据初始标记的对象所直接或间接引用的对象进行标记,比如下图对OBJ_A2,OBJ_AN进行并发标记。这个阶段并没有STW,所以可以创建对象,新增新的引用,也会让某些对象失去引用,比如下图,OBJ_B1已经变成垃圾了,OBJ_C1是新增存活的对象。这个阶段由于对老年代所有的对象进行跟踪,所以是非常耗时的。
jvm - CMS收集器

重新标记

在并发标记中,我们看到存活对象OBJ_C1等以及垃圾对象OBJ_B1等是没有被标记出来的,所以这个阶段就是对这些对象进行重新标记。
这个阶段也有STW,但是仅仅对并发标记中有变动的对象进行标记,这些数量比较少,所以速度也是很快。
jvm - CMS收集器

并发清除

这个阶段,就是在重新标记后,对垃圾对象的清理,和并发标记一样,都很耗时,由于并没有STW,所以对程序的运行影响不大。CMS采用的是标记与清除算法。
jvm - CMS收集器

缺点

CMS的4个阶段,初始标记和重新标记需要STW,但是时间短,影响不大。并发标记和并发清除不需要STW,虽然耗时,但是并发执行的,影响也不大,看起来CMS很完美,但是他也有一些缺点。

CPU

CMS默认启动的回收线程数是(CPU数量+3)/ 4,也就是当CPU在4个以上时,并发回收时垃圾收集线程不少于25%的CPU资源,并且随着CPU数量的增加而下降。比如服务器是2核4G,那就需要用(2+3)/4=1个线程去处理并发的标记和并发清除,这时候只剩下1个线程处理其他事情。

浮动垃圾

浮动垃圾的产生,主要是在并发清理阶段。重新标记后,CMS垃圾回收器会知道哪些需要清理,在并发清理阶段,他就清理重新标记后的垃圾对象,这个阶段并没有STW,所以有可能产生新的对象,比如下图的OBJ_N,创建完后,栈帧被回收,引用就没了,他在这个阶段是不能被清除的,只能等下一次垃圾回收的时候,被标记并清除。
如果这个阶段进入老年代的对象超过了剩余空间,就会出现Concurrent Mode Failure失败,那虚拟机会临时启用Serial Old收集器进行老年代的垃圾收集。
可以用XX:CMSInitiatingOccupancyFraction设置老年代空间被占用多少百分比触发CMS回收,JDK1.6后默认92%。
jvm - CMS收集器

空间碎片

jvm - 垃圾收集算法提过,标记-清除算法会产生空间碎片的,如果连续的内存空间不够存放即将进入老年代的对象,此时就会触发Full GC。
为了避免这种情况,CMS收集器提供了一个-XX:+UseCMSCompactAtFullCollection,默认打开的,当Full GC完成后,他会STW,进行内存整理,把存活的对象紧密的靠在一起,腾出连续空间。
如果每次都要重新内存,那都会STW,所以CMS还提供了-XX:CMSFullGCsBeforeCompaction参数,默认是0,他的意思是进行了多少次Full GC后才整理内存。

点赞
收藏
评论区
推荐文章
陈发良 陈发良
4年前
工作问题记录总结2021--1
1问题:在ios设备,怎么判断玩家是从通过浏览器进入,还是通过主屏幕点击进来的?解决方法:(不刷新页面,但是改变页面的url地址)1用户在第一次使用浏览器进入的时候,给url地址添加标记,2当用户添加到主屏幕的时候,就多了标记3网页打开的时候,判断是否有这个标记,如果有这个标记,就是通过主屏幕进入,如果没有这个标记,就是通过浏览器进入主
Wesley13 Wesley13
3年前
java 面试知识点笔记(六)垃圾回收 下篇
问:Object的finalize()方法的作用是否与C的析构函数作用相同?与C的析构函数不同,析构函数调用是确定的,而finalize是不确定的将未被引用的对象放置于FQueue队列(垃圾收集器确定一个对象死亡时需要至少两次标记过程。第一次是可达性分析,没有引用则会标记并且判断是否执行finalize方法,如果对象覆
Wesley13 Wesley13
3年前
JSON与XML的区别比较
1.定义介绍(1).XML定义扩展标记语言(ExtensibleMarkupLanguage,XML),用于标记电子文件使其具有结构性的标记语言,可以用来标记数据、定义数据类型,是一种允许用户对自己的标记语言进行定义的源语言。XML使用DTD(documenttypedefinition)文档类型定义来组织数据;格式统一,跨平台和语
Stella981 Stella981
3年前
Jira Epic在完成状态时,如何让Epic在Scrum面板待办事项中不显示?
遇到的问题:Epic在完成状态时,仍旧在Scrum面板待办事项中显示,需要手动将其【标记完成】而想要的效果是:Epic到完成状态时,Epic自动标记完成!(https://oscimg.oschina.net/oscnet/03911d609eda7627b34e9cf9ef0a4c25ab9.jpg)通过Chrome控制台查看【标记完成】
Wesley13 Wesley13
3年前
Unity序列化之XML,JSON
1.先介绍一下XML和JSON是什么东西吧?   (1)XML扩展标记语言(ExtensibleMarkupLanguage,XML),用于标记电子文件使其具有结构性的标记语言,可以用来标记数据、定义数据类型,是一种允许用户对自己的标记语言进行定义的源语言。XML使用DTD(documenttypedefinition
Wesley13 Wesley13
3年前
Markdown简介
Markdown是什么  Markdown是一种完全由标点符号标记的纯文本,这些标点符号被Markdown赋予表达含义以使得文本看上去可以代表特定的语义,比如:井号标记代表标题、星号标记代表强调、两个星号标记代表加粗。  Markdown设计之初只是让写作文档和阅读文档变得更容易,因Markdo
Wesley13 Wesley13
3年前
CMS垃圾回收过程
1.总体介绍:CMS(ConcurrentMarkSweep)是以牺牲吞吐量为代价来获得最短回收停顿时间的垃圾回收器。对于要求服务器响应速度的应用上,这种垃圾回收器非常适合。在启动JVM参数加上\XX:UseConcMarkSweepGC ,这个参数表示对于老年代的回收采用CMS。CMS采用的基础算法是:标记—清除。2.CMS
Wesley13 Wesley13
3年前
GC策略&内存申请、对象衰老
      JVM里的GC(GarbageCollection)的算法有很多种,如标记清除收集器,压缩收集器,分代收集器等等,详见HotSpotVMGC的种类(https://www.oschina.net/action/GoToLink?urlhttp%3A%2F%2Fwww.cnblogs.com%2Fredcreen%2Farchive
Stella981 Stella981
3年前
Python垃圾回收机制
对于Python垃圾回收机制主要有三个,首先是使用引用计数来跟踪和回收垃圾,为了解决循环引用问题,就采用标记清除的方法,标记清除的方法所带来的额外操作实际上与系统中总的内存块的总数是相关的,当需要回收的内存块越多,垃圾检查带来的额外操作就越多,为了提高垃圾收集的效率,采用“空间换时间的策略”,即使用分代机制,对于长时间没有被回收的内存就
Stella981 Stella981
3年前
JVM调优总结(三)
可以从不同的的角度去划分垃圾回收算法:按照基本回收策略分引用计数(ReferenceCounting):比较古老的回收算法。原理是此对象有一个引用,即增加一个计数,删除一个引用则减少一个计数。垃圾回收时,只用收集计数为0的对象。此算法最致命的是无法处理循环引用的问题。标记清除(MarkSweep):
Stella981 Stella981
3年前
JVM的GC算法总结
Java程序在运行过程中,会产生大量的内存垃圾(一些没有引用指向的内存对象都属于内存垃圾,因为这些对象已经失去标记,程序用不了它们了,对程序而言它们已经废弃),为了确保程序运行时的性能,java虚拟机在程序运行的过程中不断地进行自动的垃圾回收(GC),这就是我们的垃圾回收机制,关于垃圾回收我总结了一下几种:标记–清除算法(MarkSweep)