G1 垃圾收集器介绍


G1 总览

首先在内存划分上,G1垃圾收集器依然是基于分代收集的。不同的是G1收集器将整个堆划分为一个个大小相同的区块(Region),每一块的内存是连续。和分代收集算法一样,G1中每个块回充当Eden、Survivor、Eden三种角色。不同的是他们是固定的,这使得内存使用非常灵活。


1.jpg

执行垃圾收集时,和 CMS 一样,G1 收集线程在标记阶段和应用程序线程并发执行,标记结束后,G1 也就知道哪些区块基本上是垃圾,存活对象极少,G1 会先从这些区块下手,因为从这些区块能很快释放得到很大的可用空间,这也是为什么 G1 被取名为 Garbage-First 的原因。

G1收集器收集器的收集活动主要包括四种

  • 新生代垃圾收集
  • 后台收集,并发周期
  • 混合式垃圾收集
  • 以及必要时Full GC

G1收集器的堆结构


2.jpg

G1收集器的并发周期包括多个阶段,其中的一些会占用所有应用线程

Old GC / 并发标记周期

接下来是 Old GC 的流程(含 Young GC 阶段),其实把 Old GC 理解为并发周期是比较合理的,不要单纯地认为是清理老年代的区块,因为这一步和年轻代收集也是相关的。下面我们介绍主要流程:

初始标记(initial-mark):stop-the-world,它伴随着一次普通的 Young GC 发生,然后对 Survivor 区(root region)进行标记,因为该区可能存在对老年代的引用。

因为 Young GC 是需要 stop-the-world 的,所以并发周期直接重用这个阶段,虽然会增加 CPU 开销,但是停顿时间只是增加了一小部分。

扫描根引用区:扫描 Survivor 到老年代的引用,该阶段必须在下一次 Young GC 发生前结束。

这个阶段不能发生年轻代收集,如果中途 Eden 区真的满了,也要等待这个阶段结束才能进行 Young GC。

并发标记:寻找整个堆的存活对象,该阶段可以被 Young GC 中断。

这个阶段是并发执行的,中间可以发生多次 Young GC,Young GC 会中断标记过程

重新标记:stop-the-world,完成最后的存活对象标记。使用了比 CMS 收集器更加高效的 snapshot-at-the-beginning (SATB) 算法。

Oracel 的资料显示,这个阶段会回收完全空闲的区块

清理:清理阶段真正回收的内存很少。

到这里,G1 的一个并发周期就算结束了,其实就是主要完成了垃圾定位的工作,定位出了哪些分区是垃圾最多的。

混合垃圾回收周期

并发周期结束后是混合垃圾回收周期,不仅进行年轻代垃圾收集,而且回收之前标记出来的老年代的垃圾最多的部分区块。

混合垃圾回收周期会持续进行,直到几乎所有的被标记出来的分区(垃圾占比大的分区)都得到回收,然后恢复到常规的年轻代垃圾收集,最终再次启动并发周期。

FULL GC

下面我们来介绍特殊情况,那就是会导致 Full GC 的情况,也是我们需要极力避免的:

  • concurrent mode failure:并发模式失败,CMS 收集器也有同样的概念。G1 启动标记周期,但在 Mix GC 之前,老年代就被填满,这时候 G1 会放弃标记周期。,Stop-The-World进行清理空间

解决办法:

  • 增加堆空间
  • 调整并发周期,即增加后台运行线程数,让并发周期尽快结束。
  • 更早的进行并发周期,默认的是堆内存的45%被占用就进行并发周期
  • 晋升失败:并发周期结束后,是混合垃圾回收周期,伴随着年轻代垃圾收集,进行清理老年代空间,如果这个时候清理的速度小于消耗的速度,导致老年代不够用,那么会发生晋升失败

解决办法: 混合垃圾回收需要更迅速完成垃圾收集,也就是说在混合回收阶段,每次年轻代的收集应该处理更多的老年代已标记区块。

  • 疏散失败:年轻代垃圾收集的时候,如果 Survivor 和 Old 区没有足够的空间容纳所有的存活对象。这种情况肯定是非常致命的,因为基本上已经没有多少空间可以用了,这个时候会触发 Full GC 也是很合理的。

解决办法:最简单的就是增加堆大小

  • 大对象分配失败,我们应该尽可能地不创建大对象,尤其是大于一个区块大小的那种对象。

G1垃圾收集器主要调优目标就是避免并发模式失败和疏散失败,一旦发生这些失败,就会导致FULL GC,避免Full Gc就是G1垃圾收集器调优的首要任务。G1垃圾收集器调优的目标之一就是调优简单,G1收集器最主要的调优就是通过一个标志:-XX:MaxGCPauseMillis=N,使用G1收集器时,该标志的默认值为200,如果G1收集器发送停顿时间超过该值,G1垃圾收集器会尝试各种方法弥补--譬如调整新生代老年代比例、调整堆大小、更早的启动后台处理、概念晋升阈值、或者是在混合式垃圾收集周期中处理更多或者更少的老年代分区

参考文章:

《Java性能权威指南》
《深入理解Java虚拟机》
深入理解g1垃圾收集器
G1 垃圾收集器介绍
G1 垃圾收集器介绍

你可能感兴趣的:(G1 垃圾收集器介绍)