2020-03-18

理解JVM之垃圾收集器详解

目录


回到顶部

前言

垃圾收集器作为内存回收的具体表现,Java虚拟机规范并未对垃圾收集器的实现做规定,因而不同版本的虚拟机有很大区别,因而我们在这里主要讨论基于Sun HotSpot虚拟机1.6版本Update22,此虚拟机包含的收集器如下所示:


image

如图展示了7种作用于不同分代的收集器,若两个收集器之间存在连线,说明他们可以搭配使用。我们堆收集器进行比较就是为了针对具体的情况选择最合适的收集器。
回到顶部

一、Serial收集器

Serial是最基本,最早的收集器,曾是JDK1.3.1之前的虚拟机新生代唯一选择,这个收集器是单线程收集器,它不仅仅只会使用一个单线程或一条收集线程区完成垃圾收集工作,更重要的是当进行垃圾收集时,其他工作线程必须暂停,直到收集结束。下图为Serial/Serial Old收集器运行过程:


image

从JDK1.3到JDK1.7,HoSpot虚拟机开发团队一直在为消除或减少工作线程因内存回收而导致停顿的现象而努力着,从Serial收集器,到Parallel收集器,再到Concurrent Mark Sweep(CMS)用户线程停顿时间不断缩减,但此现象并未完全消除。
到目前为止,Serial是虚拟机运行在Client模式下的默认的新生代收集器,它也有很多优点:简单高效(与其他收集器单线程相比),对于限定单个CPU,由于没有线程交互开销,垃圾收集获得最高的单线程收集效率。Serial收集器是运行在Client模式下的虚拟机的好的选择。

回到顶部

二、ParNew收集器

ParNew收集器是Serial收集器的多线程版本,除使用多线程进行垃圾收集外,其余行为包括扣Serial收集器可用的控制参数、收集算法、Stop The World 、对象分配规则、回收策略等与Serial完全一样。ParNew收集器的工作过程如下图所示:


image

ParNew是运行在Server模式下虚拟机种的首选新生代收集器,因为除了Serial收集器,目前只有它能够与CMS收集器配合工作。ParNew收集器由于存在线程交互,可能不会有比Serial更好的效果但随着CPU数量的增加,它对于GC时系统资源的利用有很大好处。
此处区分以下并行和并发:
并行(Parallel):多条垃圾收集线程并行工作,此时用户线程仍处于等待状态。
并发(Concurrent):指用户线程与垃圾收集线程同时执行(不一定并行,可能交替执行),用户程序继续运行,垃圾收集程序运行在另一个CPU上。

回到顶部

三、Parallel Scavenge收集器

Parallel Scavenge收集器是一个新生代收集器,它时复制算法的收集器,也是并行的多线程收集器,它的关注点与其他收集器不同,其它收集器尽可能缩短用户线程停顿时间,Parallel Scavenge收集器则是达到一个可控制的吞吐量(吞吐量=运行用户代码时间/(运行代码时间+垃圾收集时间))。停顿时间越短越适合需要用户交互程序,良好相应速度能提升用户体验;高吞吐量则可以最高效率的利用CPU时间,尽快完成程序任务,适用于后台运算而不需要太多交互的任务。

回到顶部

四、Serial Old收集器

Serial Old是Serial的老年版本,它也是单线程收集器,使用“标记-整理“算法。它主要是被Client模式下的虚拟机使用。在Server模式下,它有两个用途:一是在JDK1.5及之前的本本种与Parallel Scavenge收集器搭配使用;二是作为CMS收集器的后备份预案,在并发收集发生Corrent Model Failure时候使用。Serial Old的工作过程如下图所示:


image

回到顶部

五、Parallel Old收集器

Parallel Old使用多线程和“标记-整理”算法,于JDK1.6开始提供,是Parallel Scavenge收集器的老年代版本。在其出现前,老年代除了Serial old外别无选择,由于Serial old的“拖累”,使用Parallel Scavenge未必能够在整体上实现吞吐量最大化的效果。Parallel Old出现后,“吞吐量”的黄金组合出现了,在注重吞吐量及CPU资源的敏感场合,优先考虑Parallel Old与Parallel Scavenge收集器组合。Parallel Old工作过程如图 所示:


image

回到顶部

六、CMS收集器

CMS(Concurrent Mark Sweep)收集器是一种以获取最短回收停顿时间为目标的收集器。CMS收集器应用于那些重视服务响应速度,系统停顿短的场景。CMS应用了“标记-清除”算法,他的运作过程包括四个场景:

  1. 初始标记(CMS initial mark)
  2. 并发标记(CMS concurrent mark)
  3. 重新标记(CMS ermark)
  4. 并发清除(CMS concurrent sweep)

由于在整个过程中耗时最长的并发标记与并发清除过程收集器线程可与用户线程一起工作,因而,可看做并发标记与并发清除的收集器线程是与用户线程一起并发执行的,其运行过程如下图所示:


image

CMS是一款优秀的收集器,他的主要优点有:并发收集,低停顿,因而也称为并发低停顿收集器,但其仍有缺点,主要表现在三个方面:

  1. CMS收集器对CPU资源非常敏感。面向并发设计的程序都对CPU比较敏感,在并发阶段,它会占用一部分线程导致应用程序变慢,降低总吞吐量。CMS默认自动回收线程数为(cpu数+3)/4,故当cpu总数小于4 时,几乎要拿出50%的资源执行线程收集器。增量式并发收集器(i-CMS)就是为解决这种情况而创造的,他的思想是:并发标记和并发清理时,让GC线程、用户线程交替运行,尽量减少GC独占资源时间,虽然使得垃圾回收时间变长,但降低了对用户程序的影响。目前,i-CMS已经不再提倡用户使用。
  2. CMS收集器无法处理浮动垃圾。当出现“Concurrent Mode Failure”导致另一次Full GC的产生。此时,CMS并发清理阶段用户线程还在运行着,无法集中处理运行程序产生的垃圾。由于垃圾收集阶段用户线程还在运行,需给线程预留内存,要是预留的内存无法满足需求,就会出现“Concurrent Mode Failure”失败,此时,临时启用Serial Old进行老年代垃圾回收,耗费时间过长。
  3. CMS算法是基于“标记-清除”的收集器,收集借书产生大量的空间碎片,当无法找到足够的连续空间存放对象就会触动Full GC。未解决这个问题,CMS增加一个碎片整理时间,此过程无法并发,停顿时间变长。

回到顶部

七、 G1收集器

G1收集器与CMS相比有两点改进:一是G1是基于“标记-整理”的收集器,不会产生空间碎片。二是他可以精确地控制停顿,能让使用者指定在一个时间片段内消耗在垃圾收集上的时间。G1可以在不牺牲吞吐量的前提下完成低停顿 内存回收。G1将java堆划分为多个大小独立的区域,并根据区域的垃圾堆积程度,维护一个优先列表,每次根据收集时间,优先回收垃圾最多的区域,保证来G1收集器在有效时间内的的回收效率。

回到顶部

附:垃圾收集器参数汇总

参数 描述
UseSerialGC 虚拟机运行在Client模式下的默认值,打开此开关,使用Serial+Serial Old收集器组合进行内存回收
UseParNewGC 打开此开关后,使用ParNew+Serial Old的收集器组合进行内存回收
UseConcMarkSweepGC 打开此开关后,使用ParNew+CMS+Serial Old的收集器组合进行内存回收。Serial Old收集器将作为CMS收集器出现Concurrent Mode Failure失败后的后备收集器使用
UseParallelGC 虚拟机运行在Server模式下的默认值,打开此开关后,使用Parallel Scavenge + Serial Old(PS MarkSweep)的收集器组合进行内存回收
UseParallelOldGC 打开此开关后,使用Parallel Scavenge + Parallel Old的收集器组合进行内存回收
SurvivorRatio 新生代中Eden区域与Survivor区域的容量比值,默认值为8,代表Eden:Survivor=8:1
PretenureSizeThreshold 直接晋升到老年代的对象大小,设置这个参数后,大于这个参数的对象将直接在老年代分配
MaxTenuringThreshold 晋升到老年代的对象年龄,每个对象在坚持过一次Minor GC之后,年龄就增加1,当超过这个参数时就进入老年代
UseAdaptiveSizePolicy 动态调整Java堆中各个区域的大小以及进入老年代的年龄
HandlePromotionFailure 是否允许分配担保失败,即老年代的剩余空间不足以应付新生代的整个Eden和Survivor区的所有对象都存活的极端情况
ParallelGCThreads 设置并行GC时进行内存回收的线程数
GCTimeRatio GC时间占总时间的比率,默认值为99,即允许1%的GC时间。仅在使用Parallel Scavenge收集器时生效
MaxGCPauseMillis 设置GC的最大停顿时间,仅在使用Parallel Scavenge收集器时生效
CMSInitingOccupancyFraction 设置CMS收集器在老年代空间被使用多少后触发垃圾收集。默认值为68%,仅在使用CMS收集器时生效
UseCMSCompactAtFullCollection 设置CMS收集器在完成垃圾收集后是否要进行一次内存碎片整理,仅在使用CMS收集器时生效
CMSFullGCsBeforeCompaction 设置CMS收集器在进行若干次垃圾收集后再启动一次内存碎片整理。仅在使用CMS收集器时生效

本文主要参考自《深入理解Java虚拟机——JVM高级特性与最佳实践》一书

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容