Java垃圾回收基础知识

最近一个运行了很久的程序出现了好几次OutOfMemory故障,造成大量业务都无法访问数据库的严重事故。事后通过加大堆内存暂时先优化了一下,顺便买了一本《Java性能权威指南》,研究了一下,发现其中的垃圾收集章节基本上能够解释所有的问题了。于是把该章节整理了一下,这些知识对于大多数的Java程序来说基本够用了。

1 垃圾收集概念

1.1 分代垃圾收集器

  1. 所有的GC算法都将堆分成了老年代和新生代。
  2. 所有的GC算法在清理新生代对象时,都使用了“时空停顿”(stop-the-world)方式

1.2 GC算法

JVM提供了4中不同的垃圾收集算法

  • Serial垃圾收集器
  • Throghput垃圾收集器
  • CMS收集器
  • G1垃圾收集器
  1. 这四种垃圾收集算法分别采用了不同的方法来缓解GC对应用程序的影响。
  2. Serial收集器常用于仅有单CPU可用以及当其他程序会干扰GC的情况(通常是默认值)。
  3. Throughput收集器在其他的虚拟机上是默认值,它能最大化应用程序的总吞吐量,但是有些操作可能遭遇较长的停顿。
  4. CMS收集器能够在应用县城运行的同时并行地对老年代的垃圾进行收集。如果CPU的计算能力足以支撑后台垃圾收集县城的运行,该算法能避免应用程序发生Full GC。
  5. G1收集器也能在应用线程运行的同时并发地对老年代进行收集,在某种程度上能够减少发生Full GC的风险。G1的设计理念使得它比CMS更不容易遭遇Full GC。

1.3 选择GC算法

GC算法的选择一方面取决于应用程序的特征,另一方面取决于应用的性能目标

GC算法与批量任务

  • 使用Throughput收集器处理应用程序线程的批量任务能最大程度地利用CPU的处理能力,通常能获得更好的性能。
  • 如果批量任务并没有使用机器上所有可能的CPU资源,那么切换到Concurrent收集器往往能取得更好的性能。

GC算法与吞吐量测试

  • 衡量标准是响应时间或吞吐量,在Throughput收集器和Concurrent收集器之间选择的依据主要是多少空闲CPU资源能用于运行后台的并发县城。
  • 通常情况下,Throughput收集器的平均响应时间比Concurrent收集器要差,但是在90%响应时间或者99%响应时间这几项指标上,Throughput收集器比Concurrent收集器要好一些。
  • 使用Throughput收集器会超负荷地进行大量Full GC时,切换到Concurrent收集器通常能获得更低的响应时间。

CMS收集器和G1收集器之间的抉择

  • 选择Concurrent收集器时,如果堆较小,推荐使用CMS收集器。
  • G1的设计使得它能够在不同的分区(Region)处理堆,因此它的扩展性更好,比CMS更易于处理超大堆的情况。

2 GC调优基础

2.1 调整堆的大小

最大堆: -Xmx
最小堆: -Xms

  1. JVM会根据其运行的机器,尝试估算合适的最大、最小堆的大小。
  2. 除非应用程序需要比默认值更大的堆,否则在进行调优时,尽量考虑通过调整GC算法的性能目标,而非微调堆的大小来改善程序性能。

2.2 代空间的调整

-XX:NewRatio=N 设置新生代与老年代的空间占用比率 , 默认值为2
-XX:NewSize=N 设置新生代的初始大小
-XX:MaxNewSize=N 设置新生代的最大大小
-XmnN 将NewSize和MaxNewSize设定为同一个值得快捷方法

Initial Young Gen Size = Initial Heap Size / ( 1 + NewRatio )

  1. 整个堆范围内,不同代的大小划分是由新生代所占用的空间控制的。
  2. 新生代的大小会随着整个堆大小的增长而增长,但这也是随着整个堆的空间比率波动变化的(依据新生代的初始值和最大值)。

2.3 永久代和元空间的调整

  1. 永久代或元空间保存着类的元数据(并非类本体的数据)。它以分离的堆的形式存在。
  2. 典型应用程序在启动后不需要载入新的类,这个区域的初始值可以依据所有类都加载后的情况设置。使得优化的初始值能够加速启动的过程。
  3. 开发中的应用服务器(或者任何需要繁重重新载入类的环境)上经常能碰到由于永久代或元空间耗尽触发的Full GC,这时老的元数据会被丢弃回收。

2.4 控制并发

-XX:ParallelGCThreads=N 控制启动的线程数
默认情况下JVM会在机器的每个CPU上运行一个线程,最多同时运行8个。一旦达到这个上限,JVM会调整算法,每超出5/8个CPU启动一个新的线程,所以总的线程数就是(N:CPU数)

ParallelGCThreads = 8 + ((N - 8) * 5/ 8)

  1. 几乎所有的垃圾收集算法中基本的垃圾回收线程数都依据机器上的CPU数目计算得出。
  2. 多个JVM运行在同一台物理机上时,依据公司计算出的县城数可能过高,必须进行优化(减少)。

2.5 自适应调整

自适应调整就是JVM会根据调优的策略不断的尝试,寻找优化性能的机会,它进行性能调优的依据是以往的性能历史:这其中隐含了一个假设,即将来GC周期的状态跟最近历史GC周期的状况可能很类似。事实也证明,在多种负荷下这一假设都是合理的,即使某个时刻内存的分配发生突变的情况,JVM也能够依据最新的情况重新调整它的大小。

自适应调整作用主要在两个方面:

  • 小型应用程序不需要为指定过大的堆而担心。
  • 很多应用程序根本不需要担心它的堆的大小,如果需要使用的堆的大小超过了平台的默认值,可以放心的分配更大的堆,不用关心其他细节,JVM会自动调整堆和代的打小,依据垃圾回收算法的性能目标,使用优化的内存量。

-XX:-UseAdaptiveSizePolicy 关闭自适应调整功能(如果堆得最大最小值相同,新生代的初始值和最大值相同时也会被关闭)

总结:

  1. JVM在堆得内部如何调整新生代及老生代的百分比是由自适应调整机制控制的。
  2. 通常情况下,我们应该开启自适应调整,因为垃圾回收算法依赖于调整后的代的大小来达到它停顿时间的性能目标。
  3. 对于已经精细调优过的堆,关闭自适应调整能获得一定的性能提升。

3 垃圾回收工具

开启GC的日志功能
-verbose:gc
-XX:+PrintGC

-XX:+PrintGCDetails 会创建更详细的GC日志(推荐使用)
-XX:+PrintGCTimeStamps或者-XX:+PrintGCDateStamps(推荐使用) 便于我们更精确地判断几次GC操作之间的时间。
两者的差距在于前者相对于0(JVM启动时间)的值,而后者是实际的日期字符串(效率稍低)。

-Xloggc:filename 指定输出到文件
-XX:+UseGCLogfileRotation -XX:NumberOfGCLogfiles=N -XX:GCLogfilesSize=N可以控制日志文件的循环。

分析日志文件的工具:GC Histogram (http://java.net/projects/gchisto)

其他工具:
jconsole: 可以实时监控堆的使用情况。

jstat: 可以实时收集数据

jstat -gcutil process_id 1000

小结:

  1. GC日志是分析GC相关问题的重要线索;我们应该开启GC日志标志(即使在生产服务器上)。
  2. 使用PrintGCDetails标志能获得更详尽的GC日志信息。
  3. 使用工具能有效地帮助我们解析和理解GC日志的内容,尤其是在堆GC日志中的数据进行归纳汇总时,它们非常有帮助。
  4. 使用jstat能动态地观察运行程序的垃圾回收操作。

4 总结

对任何一个Java应用程序而言,垃圾收集的性能都是其构成整体性能的关键一环。虽然对大多数的应用程序来说,调优的工作仅仅是选择合适的垃圾收集算法,或者在需要的时候,增大应用程序堆空间。

自使用调整让JVM能够自动地调整它的行为,使用给定的堆,提供尽可能好的性能。

更复杂的应用往往需要额外的调优,尤其是针对特定GC算法的调优。

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

推荐阅读更多精彩内容