官方文档:https://docs.oracle.com/javase/8/docs/technotes/guides/vm/gctuning/g1_gc_tuning.html
https://www.oracle.com/technetwork/cn/articles/java/g1gc-1984535-zhs.html
官方文档翻译:
混合回收
成功完成并发标记周期后, g1 gc 从执行年轻代回收切换到执行混合回收。在混合回收中, g1 gc 可选择将一些老年代区域添加到将回收的 Eden 和幸存者区域集。添加的老年区域的确切数量由几个flag控制。g1 gc 回收足够数量的老年代区域 (通过多次混合回收) 后, g1 将恢复到执行年轻代回收, 直到下一个并发标记周期完成。
我们知道年轻代回收是把年轻代活着的对象都拷贝到Survivor的特定区域(Survivor to),剩下的Eden和Survivor from就可以全部回收清理了。那么,mixed GC就是把一部分老年区的region加到Eden和Survivor from的后面,合起来称为collection set, 就是将被回收的集合,下次mixed GC evacuation把他们所有都一并清理。选old region的顺序是垃圾多的(存活对象少)优先,这也是garbage first的来历。这也是利用了G1堆逻辑分区(Region)的灵活性。G1似乎并没有把Survivor分成from和to,但是回收时的道理是一样的,暂且这么理解。
那么怎么决定把那些老年区region回收呢?这就是并发标记周期concurrent marking cycle要做的,它分为五个步骤:
(1)初始标记(initial mark,STW)
在此阶段,G1 GC 对根进行标记。该阶段主要利用了常规的年轻代垃圾回收暂停(STW)。
(2)根区域扫描(root region scan)
G1 GC 在初始标记的年轻代存活区扫描对老年代的引用,并标记被引用的对象。该阶段与应用程序(非 STW)同时运行,并且只有完成该阶段后,才能开始下一次 STW 年轻代垃圾回收。
(3)并发标记(Concurrent Marking)
G1 GC 在整个堆中查找可访问的(存活的)对象。该阶段与应用程序同时运行,可以被 STW 年轻代垃圾回收中断。
(4)最终标记(Remark,STW)
该阶段是 STW 回收,帮助完成标记周期。G1 GC 清空 SATB 缓冲区,跟踪未被访问的存活对象,并执行引用处理。
(5)清除垃圾(Cleanup,STW)
在这个最后阶段,G1 GC 执行统计和 RSet 净化的 STW 操作。在统计期间,G1 GC 会识别完全空的区域和可供进行混合垃圾回收的区域。清理阶段在将空白区域重置并添加到空闲列表时为部分并发。注意完全空的region不会被加到CSet,都在这个阶段直接回收了。
观察:
1.从(3)开始并发标记周期这个过程可以和年轻代回收并发,当然看起来年轻代回收会打断(3),(4)和(5)会暂停所有访问堆的线程(STW)。
2.混合回收就是把老年代的一些区域加在将要回收的Eden和survivor的后面,这些所有将要回收的区域叫做Collection Set(CSet),最后用年轻代回收算法一起回收。
希望可以帮到大家理解,欢迎留言探讨。