CMS和G1

CMS的回收流程

CMS的核心思想就是用并发换停顿时间,7个阶段只有两个需要STW,其他都是和应用线程一起跑,代价是cpu占用高,有内存碎片,可能并发失败

  1. 初始标记:STW,只标记GC Roots直接可达的对象,速度很快,通常几毫米
  2. 并发标记:和应用线程并发跑,从根递归扫描整个老年代,耗时最长,但不停顿
  3. 并发预清理:扫描卡表脏区域,标记新晋升对象,为重新标记分担工作
  4. 可中断预清理:等待一次Young GC ,减少年轻代对象数量,进一步减轻重新标记压力
  5. 重新标记:STW,这是CMS的瓶颈,修正并发期间变动,需要重新扫描GC Roots,年轻代,脏区域
  6. 并发清理:标记清除算法回收垃圾,不压缩,不移动,会有内存碎片
  7. 并发重置:重置状态,为下一次回收做准备
它的优缺点

优点:低停顿,响应快

  • 缺点: 标记清除算法导致的内存碎片
    • 并发阶段占用cpu,影响吞吐量
    • 需要预留空间,老年代满了就会触发Concurrent Mode Failure,退化成串行 Full GC
    • 浮动垃圾(并发清理时新产生的垃圾只能下次处理)

G1是什么

G1从jdk9之后成为默认GC,它的设计目的是用可控的停顿时间内,尽量高的吞吐量

它不再像传统GC把堆划分为连续的新生代,老年代,而是切成几百上千个大小相等的小格子,没有物理上的分界,Eden、Survivor、老年代都是 Region 的逻辑集合,数量可以动态调整.并且它是全堆管理,每次GC时它会优先记录每个Region的垃圾比例,每次GC时它挑选垃圾最多的几个Region优先回收

G1的流程

分为两大块:并发标记和对象拷贝
并发标记 :基于SATB算法,分四个阶段

  • 初始标记:通常搭便车,在YoungGC的STW后完成,标记GC Roots字节引用的对象
  • 并发标记:目的和G1相似都是遍历对象图,标记存活对象,但核心机制是SATB快照算法(记录谁丢失了引用,保证快照中的对象不会被回收)
  • 最终标记:STW,处理SATB队列积压的引用变更记录,修正并发期间的对象的变动,将漏掉的对象补标,确保标记结果的准确性
  • 清理阶段:STW,基于bitmap统计每个Region的存活对象,若全空则立即回收,计算回收价值,更新CSET候选列表
    对象拷贝 :G1回收一个Region时,不会原地清理垃圾,而是把这个Region存活的对象拷贝到另一个空的Region,然后直接把原Region清空
    这个阶段是STW的,根据标记结果和回收价值,选一批Region组成Collection Set,然后把CSet里的存活对象拷贝到新的Region。它的性能瓶颈就在此:存活对象越多,拷贝时间越长,G1会优先选择垃圾多的Region回收
    G1的两个回收模式
    Young GC:只回收Eden和Survivor的Region,触发条件是Eden区满了
    Mixed GC:会回收所有的年轻代,和一部分老年代,这个参数默认是45%,分多次回收老年代,每次挑垃圾比例高的Region清理
    什么叫"漏标"?哪个算法解决了它?
    "漏标"是指一个本应存活的对象被错误标记成垃圾。CMS 和 G1 都解决了漏标问题,但方式不同:
  • CMS(增量更新):记录"谁引用了新对象",重新标记阶段重新扫描。
  • G1(SATB):记录"谁丢失了引用",保证快照中的对象不会被回收
相关推荐
被子你放开我15 分钟前
CRMEB PHP多商户升级4.0太麻烦了
开发语言·php
阿里嘎多学长16 分钟前
2026-06-01 GitHub 热点项目精选
开发语言·程序员·github·代码托管
JAVA社区30 分钟前
Java高级全套教程(十一)—— Kubernetes 超详细企业级实战详解
java·运维·微服务·容器·面试·kubernetes
醒醒该学习了!30 分钟前
Anaconda安装教程+第一个python例子
开发语言·python
在繁华处2 小时前
Java从零到熟练(九):并发编程基础
java·开发语言
木头程序员2 小时前
SSM框架学习笔记
java·开发语言·mysql·spring·maven
李白你好2 小时前
页面资产梳理 · 技术指纹识别 · Spring 端点探测
java·后端·spring
一起逃去看海吧2 小时前
dify-03
java·linux·开发语言
我是一颗柠檬2 小时前
【Java后端技术亮点】热Key探测与本地缓存二级防护:Redis热点问题的终极解决方案
java·redis·后端·缓存·中间件