Go1.6与JVM CMS的垃圾回收对比

自从Go1.5引入了真正的并发GC后, Go1.6进一步进行了优化,使得Go在上百G级的堆大小时依然能将STW时间控制在20ms以内:
这里写图片描述

而Java8的G1收集器,默认参数下在100G以上的heap下,会造成秒级的STW。虽然可以通过-XX:MaxGCPauseMillis调整,但是是以牺牲大量吞吐量为代价。这里浅析一下Go能做到比G1更短的STW的原因。

轮流挂起协程

JVM的CMS收集器在工作时,大致分为4个阶段:

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

其中1, 3 是需要STW的阶段,CMS的停顿也是由这2个阶段引发的。Go1.5中的CMS也分为这些阶段,其中1, 3同样需要STW。那为什么Go会停顿时间更少呢?原因是,Go的CMS在第3阶段并不是挂起所有goroutine,而是轮流挂起。如此一来,3阶段就不会造成整个程序的停顿,从而就没有算入到STW时间之中。

Go触发GC的时机

Go的gc触发条件也与JVM的gc有很大区别。JVM通常是堆的使用到达某一阀值,或发生new操作失败时gc。而Go则是当从上次gc以来,新创建的对象大小等于上次gc以后存活下来的对象时触发gc. 这样,每次gc的压力就不会像JVM那么大,STW时间理所当然会短很多,但也牺牲了吞吐量。

Go比Java产生更少的内存垃圾

Go的对象(即struct类型)是可以分配在栈上的。Go会在编译时做静态逃逸分析(Escape Analysis), 如果发现某个对象并没有逃出当前作用域,则会将对象分配在栈上而不是堆上,从而减轻了GC压力。其实JVM也有逃逸分析,但与Go不同

### JVM CMS垃圾回收器的标记清除流程 CMS(Concurrent Mark-Sweep)垃圾回收器是一种以最小化停顿时间为目标的老年代垃圾回收器。它的核心思想是在应用线程运行的同时完成大部分垃圾回收工作,从而减少因垃圾回收导致的应用暂停时间。以下是CMS垃圾回收器中标记清除的具体流程: #### 初始标记阶段 初始标记阶段是一个短暂的STW(Stop-The-World)操作,用于标记老年代中可以直接从根集合访问到的对象。此阶段的时间较短,因为它只需要扫描GC Roots并标记直接关联的对象[^1]。 #### 并发标记阶段 在并发标记阶段,CMS会遍历整个对象图来查找所有可到达的对象。这一阶段允许应用程序线程继续运行,因此不会造成明显的停顿。然而,由于用户线程在此期间仍然活动,可能导致某些对象的状态发生变化,这会在后续阶段进行修正[^2]。 #### 重新标记阶段 重新标记阶段是对并发标记阶段的结果进行校正的过程。在这个阶段,CMS会对那些在并发标记期间状态发生改变的对象再次进行标记。为了确保准确性,这个阶段也会引入一定的STW,但它持续的时间通常比初始标记稍长一些,但仍远远小于并发标记所需的时间[^5]。 #### 清理阶段 清理阶段负责实际释放未被标记的对象所占用的空间。CMS采用的是“标记-清除”算法的一部分逻辑,在此阶段中,它会删除之前已经被标记为不可达的所有对象,并更新可用空间的信息。需要注意的是,“标记-清除”算法的一个显著问题是容易产生内存碎片[^3][^4]。 #### 并发重置阶段 最后一步是并发重置阶段,此时系统准备下一轮新的垃圾收集周期所需的资源和数据结构,比如清空记忆集中的记录等信息以便于下次使用[^2]。 ```java // 下面展示了一个简单的模拟CMS GC过程伪代码表示: public class CMSSimulation { public static void main(String[] args) throws InterruptedException { Object root = new Object(); // 假设这是我们的Root对象 markInitial(root); // Initial Mark Phase (STW) Thread.sleep(100); // Simulate Concurrent Execution Time concurrentMark(root); // Concurrent Marking Phase reMarkPhase(root); // Remark Phase (Short STW Again) sweepUnreachableObjects(); // Clear Unmarked Objects During Sweep Phase resetDataStructures(); // Reset Data Structures For Next Cycle } private static void markInitial(Object obj){ System.out.println("Performing initial marking..."); } private static void concurrentMark(Object obj){ System.out.println("Starting concurrent marking phase..."); } private static void reMarkPhase(Object obj){ System.out.println("Re-mark objects that changed during the concurrent phase."); } private static void sweepUnreachableObjects(){ System.out.println("Sweep and clear unreachable objects from memory."); } private static void resetDataStructures(){ System.out.println("Resetting internal data structures for next cycle."); } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值