Go 原理之 gc 垃圾回收机制:三色标记 + 混合写屏障(需要 STW)
Go 原理之 gc 垃圾回收机制
一、常见垃圾回收算法
| 垃圾回收算法 | 描述 | 代表语言 | 优缺点 |
|---|---|---|---|
| 引用计数 | 为每个对象维护一个引用计数,记录对象被引用的次数 每当一个对象被引用时,引用计数就会增加。 当对象不再被引用时,引用计数就会减少。 如果对象的引用计数变为 0, 则对象可以被垃圾回收器回收 |
Python、PHP | 优点: 实现简单,处理快 缺点: 无法处理循环引用,两个对象相互引用,计数永远不为0 |
| 分代收集 | 按照对象生命周期长短划分不同的代空间, 生命周期长的放入老年代,短的放入新生代, 不同代有不同的回收算法和回收频率 |
Java | 优点: 性能好 缺点: 需要 STW,算法复杂 |
| 三色标记法 | 从根变量开始遍历所有引用的对象,标记引用的对象为不同颜色, 被标记为白色的对象进行回收 |
Golang | 优点: 解决了引用计数的缺点 缺点: 需要 STW,暂时停掉程序运行 |
注意️:以上都需要 STW
二、Go 的 gc:三色标记 + 混合写屏障
(一)三色标记
v1.13之前,go 使用的是 标记-清除法,需要 stw ,效率极低;
v1.15之后,go 采用 三色标记 + 混合写屏障 极大的降低stw的时间,提高gc性能
三色标记:白色(清除对象) + 灰色(过渡对象,受保护, 最终变黑色) + 黑色(受保护)
可达对象引用关系举例
可达的意思就是可以关联到的,有对象引用它了
对象1 = 对象2 // 对象2可达,对象1引用了对象2,对象2 被 对象1 引用
对象1 = 对象3
对象2 = 对象3
对象2 = 对象5
三色标记-流程
- 初始时,所有对象被标记为白色
- gc 开始,遍历 rootset 根节点,将有引用对象的对象标记为 灰色,存入灰色对象列表
- 遍历 灰色对象,将直接可达对象标记为 灰色,并将自身标记为 黑色
- 重复第3步,直到标记完所有的对象 (灰色对象列表为空)
- 将白色对象清除,保留黑色对象

(二)混合写屏障
三色标记存在并发问题:
在三色标记期间,如果没有STW,并发创建对象,可能存在 == 垃圾对象或误删对象 == 的情况:
黑色对象的引用对象被删除,则不可达,正常黑色对象应该被回收,但是gc期间只会循环遍历灰色列表,不会回收黑色对象,因此该对象为垃圾对象(多余垃圾对象)eg:对象1已经被标记为黑色,表示该对象有引用方,受保护,如果没有stw,该对象的引用可能被删除,正常应该转为白色对象被清除,然gc并不会清除黑色对象
黑色对象引用了白色对象,白色对象有了引用对象应该被保护,但仍然被无情的回收(清掉不该清的对象)
白色对象只有被灰色对象引用情况,才会判断是否需要清理,白色对象如果在gc期间引用了黑色对象,那只会被误删除
所以 go 引入了 混合写屏障 机制,满足:
- 强三色不变式:黑色对象不允许引用了白色对象;因为一旦引用,该黑色对象将不会继续参与 gc,白色对象会被无理清除
- 弱三色不变式:黑色对象可以引用白色对象,但该白色对象必须被其它灰色对象或其上游有灰色对象引用,否则该白色对象将被无理清除
这里需要注意一点,插入屏障仅会在堆内存中生效,不对栈内存空间生效,这是因为 go 在并发运行时,大部分的操作都发生在栈上,函数调用会非常频繁。
数十万goroutine的栈都进行屏障保护自然会有性能问题
所以 gc 期间,任何在栈上新创建的对象,均为黑色。
| 混合写屏障 | 开启期间 | 描述 |
|---|---|---|
| 插入写屏障 | 创建的新对象为灰色对象 | 满足:强三色不变式。 不会存在黑色对象引用白色对象 |
| 删除写屏障 | 被删除的对象,如果自身为灰色或者白色,那么被标记为灰色 | 满足:弱三色不变式 (保护灰色对象到白色对象的路径不会断) |
(三)优缺点
优点:
减少stw时间,三色标记需要stw整个程序,混合写屏障(分段stw)可以有效降低stw的时间
缺点:
回收精度低,有些垃圾需要在下一轮 gc 清理
(四)完整的 gc 流程
三色标记 + 混合写屏障
- 标记准备(Mark Setup):开启混合写屏障(Write Barrier),需 STW(stop the world)
- 标记开始(Marking):使用三色标记法并发标记 ,与用户程序并发执行
- 标记终止(Mark Termination):对触发写屏障的对象进行重新扫描标记,关闭写屏障(Write Barrier),需 STW(stop the world)
- 清理(Sweeping):将需要回收的内存归还到堆中,将过多的内存归还给操作系统,与用户程序并发执行
三、源码解析
/go/1.18.3/libexec/src/runtime/mgc.go
四、常见问题
1. gc 多久执行一次,什么时候触发
- 定时触发:Go 运行时系统会根据一定的时间间隔定期触发垃圾回收。时间间隔根据程序的内存使用情况和性能需求进行自适应调整
- 内存分配触发:当程序申请的内存超过一定阈值时,Go 运行时会触发垃圾回收,以防止过度使用内存
- 栈伸缩触发:当 Goroutine 的栈空间不足以容纳当前的执行需要时,Go 运行时会触发垃圾回收来扩展栈空间
- 主动触发:调用 runtime.GC
- 空间不足时触发: 当前线程的内存管理单元中不存在空闲空间时,创建32KB以下的对象可能触发垃圾收集,创建32KB以上的对象时,一定会尝试触发
2. 为什么混合写屏障不保护栈的引用
因为go在并发运行时,大部分的操作都发生在栈上,函数调用会非常频繁。数十万goroutine的栈都进行屏障保护自然会有性能问题
虽然混合写屏障不保护栈上的引用,但 Go 语言的垃圾回收器在标记终止阶段会对栈进行重新扫描。在这个阶段,会暂停所有的用户程序(STW),对栈上的对象和引用进行精确的标记,确保所有可达对象都被正确标记。这样就弥补了不使用写屏障保护栈上引用的不足,保证了垃圾回收的正确性。
综上所述,混合写屏障不保护栈的引用是为了在保证垃圾回收正确性的前提下,尽可能提高程序的性能和降低实现复杂度。通过栈重新扫描机制,也能确保栈上的可达对象不会被错误回收。
3. gc 过程中那一部分使用了 STW
标记准备阶段(Mark Setup)
在这个阶段,垃圾回收器需要初始化标记状态,开启写屏障等操作。为了确保标记的正确性,需要暂停所有的用户程序,进行 STW
在标记阶段开始前会进行一次STW,暂停所有goroutine的执行,然后再进行标记操作
标记终止阶段(Mark Termination)
在并发标记阶段结束后,可能还有一些标记工作没有完成,如一些新创建的对象或者修改的引用关系没有被标记。因此,需要暂停所有的用户程序,完成剩余的标记工作,关闭写屏障,统计所有需要回收的对象。
原文地址
Go 原理之 gc 垃圾回收机制:三色标记 + 混合写屏障(需要 STW)
Go 原理之 gc 垃圾回收机制:三色标记 + 混合写屏障(需要 STW)的更多相关文章
- 面试官,不要再问我“Java GC垃圾回收机制”了
Java GC垃圾回收几乎是面试必问的JVM问题之一,本篇文章带领大家了解Java GC的底层原理,图文并茂,突破学习及面试瓶颈. 楔子-JVM内存结构补充 在上篇<JVM之内存结构详解> ...
- java面试题之----JVM架构和GC垃圾回收机制详解
JVM架构和GC垃圾回收机制详解 jvm,jre,jdk三者之间的关系 JRE (Java Run Environment):JRE包含了java底层的类库,该类库是由c/c++编写实现的 JDK ( ...
- 乐字节Java|GC垃圾回收机制、package和import
本文接上一篇:乐字节Java|this关键字.static关键字.block块.本文是接着讲述JavaGC垃圾回收机制.package 和 import语句. 一.GC垃圾回收机制 GC全名:Garb ...
- 通俗易懂.NET GC垃圾回收机制(适用于小白面试,大牛勿喷)
情景:你接到xx公司面试邀请,你怀着激动忐忑的心坐在对方公司会议室,想着等会的技术面试.技术总监此时走来,与你简单交谈后.... 技术:你对GC垃圾回收机制了解的怎么样? 你:还行,有简单了解过. 技 ...
- 精华推荐 | 【JVM深层系列】「GC底层调优系列」一文带你彻底加强夯实底层原理之GC垃圾回收技术的分析指南(GC原理透析)
前提介绍 很多小伙伴,都跟我反馈,说自己总是对JVM这一块的学习和认识不够扎实也不够成熟,因为JVM的一些特性以及运作机制总是混淆以及不确定,导致面试和工作实战中出现了很多的纰漏和短板,解决广大小伙伴 ...
- JVM和GC垃圾回收机制和内存分配
JVM运行期间 线程共享 线程私有 线程共享: 方法区 堆方法区:存放可以共享数据,静态常量,类的共有方法属性字段等,可以共享的存在方法区. 堆:存放class对象 . 线程私有:本地方法栈 虚拟机栈 ...
- JVM架构和GC垃圾回收机制
深入理解系列之JDK8下JVM虚拟机(1)——JVM内存组成 https://blog.csdn.net/u011552404/article/details/80306316 JVM架构和GC垃圾回 ...
- GC垃圾回收机制详解
JVM堆相关知识 为什么先说JVM堆? JVM的堆是Java对象的活动空间,程序中的类的对象从中分配空间,其存储着正在运行着的应用程序用到的所有对象.这些对象的建立方式就是那些new一类的操作 ...
- asp.net 之 GC (垃圾回收机制)
今天抽时间好好整理了下GC相关知识,看了CSDN和博客园的几篇文章,有了一定的简单了解,决定根据个人理解整合一份随笔写下来,望诸位指教. 一:基础问题 1.首先需要知道了解什么是GC? GC如其名,就 ...
- Android内存优化3 了解java GC 垃圾回收机制1
开篇废话 如果我们想要进行内存优化的工作,还是需要了解一下,但这一块的知识属于纯理论的,有可能看起来会有点枯燥,我尽量把这一篇的内容按照一定的逻辑来走一遍.首先,我们为什么要学习垃圾回收的机制,我大概 ...
随机推荐
- 赣CTF-Misc方向wp
checkin 下载附件,一张图片,拖进010,在文件尾看到隐藏文本,提取并用社会主义价值解密 ez_forensics 提示为结合题目进行想象,我们会想到取证第一步vc挂载,但是需要密码,研究图片, ...
- AI智能体策略FunctionCalling和ReAct有什么区别?
Dify 内置了两种 Agent 策略:Function Calling 和 ReAct,但二者有什么区别呢?在使用时又该如何选择呢?接下来我们一起来看. 1.Function Calling Fun ...
- 腾讯IMA VS 飞书知识问答:谁才是2025最强AI知识库?
AI创业失败,可私聊经验教训分享... 前几天小伙伴在讨论我开发的一套社群运营AI分身,其本质其实是一套个人知识库的AI产品,其依赖的就是我过往发布的文章. 这类AI聊天分身,最简单.不考虑" ...
- 「Log」做题记录 2024.1.1-2024.1.28
\(2024.1.1-2024.1.7\) \(\color{blueviolet}{P1501}\) LCT 板子,链加链乘查询链和,断边加边. \(\color{black}{P4332}\) L ...
- 【中英】【吴恩达课后编程作业】Course 4 -卷积神经网络 - 第四周作业
[中文][吴恩达课后编程作业]Course 4 - 卷积神经网络 - 第四周作业 - 人脸识别与神经风格转换 上一篇:[课程4 - 第四周测验]※※※※※ [回到目录]※※※※※下一篇:[待撰写-课程 ...
- 图神经网络(GNN)模型的基本原理
一.概述 在人工智能领域,数据的多样性促使研究人员不断探索新的模型与算法.传统的神经网络在处理像图像.文本这类具有固定结构的数据时表现出色,但面对具有不规则拓扑结构的图数据,如社交网络.化学分子结 ...
- .net 实现 CQRS 的的一个设想
一.先总结一下以前的经验 本人前公司项目开发基本算是CQRS的 后台和数据接口用.net(后大部分转java),前台nodejs 数据统计用hive 但是涉及列表含模糊搜索基本都是交给站内搜索团队,他 ...
- HarmonyOS运动开发:深度解析文件预览的正确姿势
鸿蒙核心技术##运动开发##Core File Kit(文件基础服务)##Preview Kit(文件预览服务)# 在 HarmonyOS 开发中,文件预览功能是一个常见的需求,尤其是在处理用户上传的 ...
- 控制ERP物料主数据通过Middleware传往CRM
先说一下优化过滤的必要性. CRM物料主数据一百多万. 感谢MDM或者相关的系统,每天通过接口更新的不知道什么东西,每天数百万的物料更新队列进入CRM. CRM系统被搞死了好几次. 然后各种优化报表, ...
- DTALK直播预约 | 数据资产管理:金融机构数据价值释放的必经之路
当前,数据对金融机构业务和发展的重要性日益凸显,释放数据生产力已经成为金融机构进行全面数字化转型的核心,这就要求金融机构以数据资产为纲不断提升自身数据资产管理能力. 本期DTALK我们邀请到雅拓信息解 ...