一文详解go底层原理之垃圾回收

2024-09-07 12:28

本文主要是介绍一文详解go底层原理之垃圾回收,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 前置知识

1.1 三色回收法

三色回收法在gov1.5版本时是主流的gc方式

简单介绍一下流程:

  1. 暂停程序执行流程(开启STW)
  2. 将新创建的对象全部标记为白色
  3. 从根节点开始遍历,把遍历到的第一层全部改为灰色
  4. 遍历一次灰色集合,将灰色集合引用对象变为黑色
  5. 重复上述步骤,知道没有灰色对象
  6. 清除白色对象
  7. 结束STW

1.2 STW

上述1.1所说的STW就是指的stop the world,简单的说就是让程序停止,肯定是要耗费大量的时间,所以我们应该尽量减少STW

因此我们来看一个如果不进行STW就进行三色回收的场景
一个白色对象被黑色对象引用,且灰色对象与它可达的白色对象断联

这时候白色对象和其下游对象可能被清理,造成对象丢失

1.3强弱不等式

强不等式:
强制性的不允许黑色对象引用白色对象

弱不等式:
黑色对象可以引用白色对象,但是这个白色对象必须存在其他灰色对象对它的引用,或者可达其链路上存在灰色对象

1.4两种屏障方式

插入写屏障 :
是在垃圾回收器执行过程中插入的一种机制,用来确保对对象的修改不会干扰垃圾回收的正确性。当程序在执行时,可能会修改对象的字段,这些字段可能指向其他对象。在这种情况下,写屏障会插入到代码中,以便在修改对象的字段时,垃圾回收器能够及时更新它的状态,从而避免垃圾回收器丢弃仍在使用的对象。
简单的说,就是如果B挂在A的下游,则B必须为灰色

但是注意,插入写屏障在栈上不能生效,原因就是栈上函数调用弹出频繁使用,所以插入写屏障过于影响性能

删除写屏障:
删除写屏障 是指在不再需要写屏障时,移除写屏障的操作。在某些情况下,垃圾回收器不再需要持续跟踪某些对象的引用,写屏障可以被删除以优化性能。删除写屏障可以减少运行时开销,提高程序的效率。

简单的说,被删除的对象,如果自身为灰色或者白色,则为灰色,这种方式回收精度过低,原因就是一个对象即使被删除掉最后一个指向它的指针,它依然可以存活一轮

2 混合写屏障+三色回收法

Go V1.8版本引入了混合写屏障机制(hybrid write barrier),避免了对栈re-scan的过程,极大的减少了STW的时间。结合了两者的优点。

注意混合写屏障是GC的一种屏障机制,所以只是当程序执行GC的时候,才会触发这种机制

​Golang中的混合写屏障满足弱三色不变式,结合了删除写屏障和插入写屏障的优点,只需要在开始时并发扫描各个goroutine的栈,使其变黑并一直保持,这个过程不需要STW,而标记结束后,因为栈在扫描后始终是黑色的,也无需再进行re-scan操作了,减少了STW的时间。

步骤
GC开始将栈上的对象全部扫描并标记为黑色(之后不再进行第二次重复扫描,无需STW)
GC期间,任何在栈上创建的新对象,均为黑色
被删除的对象标记为灰色
被添加的对象标记为灰色
满足->变形的弱三色不变式

这里我们注意, 屏障技术(步骤三四)是不在栈上应用的,因为要保证栈的运行效率

这篇关于一文详解go底层原理之垃圾回收的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!


原文地址:
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.chinasem.cn/article/1145055

相关文章

一文带你搞懂Python中__init__.py到底是什么

《一文带你搞懂Python中__init__.py到底是什么》朋友们,今天我们来聊聊Python里一个低调却至关重要的文件——__init__.py,有些人可能听说过它是“包的标志”,也有人觉得它“没... 目录先搞懂 python 模块(module)Python 包(package)是啥?那么 __in

Java实现优雅日期处理的方案详解

《Java实现优雅日期处理的方案详解》在我们的日常工作中,需要经常处理各种格式,各种类似的的日期或者时间,下面我们就来看看如何使用java处理这样的日期问题吧,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言一、日期的坑1.1 日期格式化陷阱1.2 时区转换二、优雅方案的进阶之路2.1 线程安全重构2

Java中的JSONObject详解

《Java中的JSONObject详解》:本文主要介绍Java中的JSONObject详解,需要的朋友可以参考下... Java中的jsONObject详解一、引言在Java开发中,处理JSON数据是一种常见的需求。JSONObject是处理JSON对象的一个非常有用的类,它提供了一系列的API来操作J

HTML5中的Microdata与历史记录管理详解

《HTML5中的Microdata与历史记录管理详解》Microdata作为HTML5新增的一个特性,它允许开发者在HTML文档中添加更多的语义信息,以便于搜索引擎和浏览器更好地理解页面内容,本文将探... 目录html5中的Mijscrodata与历史记录管理背景简介html5中的Microdata使用M

html5的响应式布局的方法示例详解

《html5的响应式布局的方法示例详解》:本文主要介绍了HTML5中使用媒体查询和Flexbox进行响应式布局的方法,简要介绍了CSSGrid布局的基础知识和如何实现自动换行的网格布局,详细内容请阅读本文,希望能对你有所帮助... 一 使用媒体查询响应式布局        使用的参数@media这是常用的

HTML5表格语法格式详解

《HTML5表格语法格式详解》在HTML语法中,表格主要通过table、tr和td3个标签构成,本文通过实例代码讲解HTML5表格语法格式,感兴趣的朋友一起看看吧... 目录一、表格1.表格语法格式2.表格属性 3.例子二、不规则表格1.跨行2.跨列3.例子一、表格在html语法中,表格主要通过< tab

Linux之计划任务和调度命令at/cron详解

《Linux之计划任务和调度命令at/cron详解》:本文主要介绍Linux之计划任务和调度命令at/cron的使用,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录linux计划任务和调度命令at/cron一、计划任务二、命令{at}介绍三、命令语法及功能 :at

Java使用SLF4J记录不同级别日志的示例详解

《Java使用SLF4J记录不同级别日志的示例详解》SLF4J是一个简单的日志门面,它允许在运行时选择不同的日志实现,这篇文章主要为大家详细介绍了如何使用SLF4J记录不同级别日志,感兴趣的可以了解下... 目录一、SLF4J简介二、添加依赖三、配置Logback四、记录不同级别的日志五、总结一、SLF4J

Java使用ANTLR4对Lua脚本语法校验详解

《Java使用ANTLR4对Lua脚本语法校验详解》ANTLR是一个强大的解析器生成器,用于读取、处理、执行或翻译结构化文本或二进制文件,下面就跟随小编一起看看Java如何使用ANTLR4对Lua脚本... 目录什么是ANTLR?第一个例子ANTLR4 的工作流程Lua脚本语法校验准备一个Lua Gramm

一文详解如何在Python中从字符串中提取部分内容

《一文详解如何在Python中从字符串中提取部分内容》:本文主要介绍如何在Python中从字符串中提取部分内容的相关资料,包括使用正则表达式、Pyparsing库、AST(抽象语法树)、字符串操作... 目录前言解决方案方法一:使用正则表达式方法二:使用 Pyparsing方法三:使用 AST方法四:使用字