Skip to content

Latest commit

 

History

History
160 lines (120 loc) · 8.77 KB

File metadata and controls

160 lines (120 loc) · 8.77 KB

垃圾回收器:基础知识

[TOC]

Go 运行时的垃圾回收器主要包含四个阶段:

  1. 准备阶段:GC 开始前的准备工作,需要 STW
  2. 标记阶段:GC 准备工作已经完成,并发的与用户代码执行,不需要 STW
  3. 终止阶段:标记工作结束,需要 STW
  4. 清扫阶段:清理标记的内存,调整触发下一次 GC 的时机,并发的与用户代码执行,不需要 STW

并发三色回收一瞥

下面四张图清晰的展示了 Go 语言内建的垃圾回收的进化历史。

在 Go 1 的时代,尽管所有的用户代码都是并发执行的,但是一旦垃圾回收器开始进行垃圾回收工作时,所有的用户代码都会停止执行,而且垃圾回收器仅在一个线程上执行,这时是最原始的垃圾回收器的实现,即单线程版的三色标记清扫。

图片截取自 [Hiltner 2017]

在 Go 1.1 时候,官方成功将三色标记清扫算法的垃圾回收的代码得以并行,从而成功缩短了用户代码的停止时间,但是这仍然会造成大量的空隙,如果用户代码是一个 Web 应用,且正在处理一个非常重要的请求时,则会对请求延迟造成巨大的影响。

图片截取自 [Hiltner 2017]

为了解决这一问题,官方使用写屏障技术开始让垃圾回收与用户代码得以并行执行。从而只有在执行写屏障和很短一段时间内需要进行 STW。

图片截取自 [Hiltner 2017]

最后在 1.8 时,通过引入混合屏障,Go 团队成功将 STW 进一步缩短,几乎解决了 STW 的问题。

图片截取自 [Hiltner 2017]

原始的标记清扫算法

Go 的垃圾回收器基于标记清扫的思想,将回收过程分为两个阶段:

  1. 追踪:从根集合(寄存器、执行栈、全局变量)开始遍历对象图,标记遇到的每个对象;
  2. 清扫:检查堆中每一个对象,将所有未标记的对象当做垃圾进行回收。

标记算法的伪代码可以按如下思路来理解:

func markFromRoots() {
    worklist.Init() // 初始化标记 work 列表
    for fld := range Roots { // 从根开始扫描
        ref := *fld
        // 标记每个遇到的对象
        if ref != nil && !isMarked(ref) {
            setMarked(ref)
            worklist.Add(ref)
            mark()
        }
    }
}
func mark() {
    for !worklist.Empty() {
        ref := worklist.Remove() // ref 已经标记过
        for fld := range Pointers(ref) {
            child := *fld
            if child != nil && !isMarked(child) {
                setMarked(child)
                worlist.Add(child)
            }
        }
    }
}

清扫算法:

func sweep(start, end) {
    // 检查堆区间内所有的对象
    for scan := start; scan < end; scan = scan.Next {
        if isMarked(scan) {
            unsetMarked(scan)
        } else {
            free(scan) // 将未标记的对象释放
        }
    }
}

内存模型

语言的内存模型定义了并行状态下拥有确定读取和写入的时序的条件。 Go 的 goroutine 采取并发的形式运行在多个并行的线程上, 而其内存模型就明确了 对于一个 goroutine 而言,一个变量被写入后一定能够被读取到的条件。 在 Go 的内存模型中有事件时序的概念,并定义了 happens before ,即表示了在 Go 程序中执行内存操作的一个偏序关系。

我们不妨用 < 表示 happens before,则如果事件 e1 < e2,则 e2 > e1。 同样,如果 e1e2 且 _e1 ≤ e2,则 e1e2 happen concurrently (e1 = e2)。 在单个 goroutine 中,happens-before 顺序即程序定义的顺序。

我们稍微学院派的描述一下偏序的概念。 (严格)偏序在数学上是一个二元关系,它满足自反、反对称和传递性。happens before(<)被称之为偏序,如果满足这三个性质:

  1. (反自反性)对于 ∀_e1_∈{事件},有:非 e1 < e1;
  2. (非对称性)对于∀_e1_, _e2_∈{事件},如果 e1 ≤ e2,e2 ≤ e1 则 e1 = e2,也称 happens concurrently;
  3. (传递性)对于∀_e1_, e2, e3 ∈{事件},如果 e1 < e2,e2 < e3,则 e1 < e3。

可能我们会认为这种事件的发生时序的偏序关系仅仅只是在探讨并发模型,跟内存无关。 但实际上,它们既然被称之为内存模型,就是因为它们与内存有着密切关系。 并发操作时间偏序的条件,本质上来说,是定义了内存操作的可见性。

编译器和 CPU 通常会产生各种优化来影响程序原本定义的执行顺序,这包括:编译器的指令重排、 CPU 的乱序执行。 除此之外,由于缓存的关系,多核 CPU 下,一个 CPU 核心的写结果仅发生在该核心最近的缓存下, 要想被另一个 CPU 读到则必须等待内存被置换回低级缓存再置换到另一个核心后才能被读到。

Go 中的 happens before 有以下保证:

  1. 初始化:main.init < main.main
  2. goroutine 创建: go < goroutine 开始执行
  3. goroutine 销毁: goroutine 退出 = ∀ e
  4. channel: 如果 ch 是一个 buffered channel,则 ch<-val < val <- ch
  5. channel: 如果 ch 是一个 buffered channel 则 close(ch) < val <- ch & val == isZero(val)
  6. channel: 如果 ch 是一个 unbuffered channel 则,ch<-val > val <- ch
  7. channel: 如果 ch 是一个 unbuffered channel 则,len(ch) == C => 从 channel 中收到第 k 个值 < k+C 个值得发送完成
  8. mutex: 如果对于 sync.Mutex/sync.RWMutex 的锁 l 有 n < m, 则第 n 次调用 l.Unlock() < 第 m 次调用 l.Lock() 的返回
  9. mutex: 任何发生在 sync.RWMutex 上的调用 l.RLock, 存在一个 n 使得 l.RLock > 第 n 次调用 l.Unlock,且与之匹配的 l.RUnlock < 第 n+1 次调用 l.Lock
  10. once: f() 在 once.Do(f) 中的调用 < once.Do(f) 的返回.

TODO: 谈及内存模型与实现的 barrier 之间的关系

编译标志 go:nowritebarriergo:nowritebarrierrecgo:yeswritebarrierrec

如果一个函数包含写屏障,则被 go:nowritebarrier 修饰的函数触发一个编译器错误,但它不会抑制写屏障的产生,只是一个断言。 go:nowritebarrier 主要适用于在没有写屏障会获得更好的性能,且没有正确性要求的情况。 我们通常希望使用 go:nowritebarrierrec

如果声明的函数或任何它递归调用的函数甚至于 go:yeswritebarrierrec 包含写屏障,则 go:nowritebarrierrec 触发编译器错误。

逻辑上,编译器为每个函数调用添加 go:nowritebarrierrec 且当遭遇包含写屏障函数的时候产生一个错误。 go:yeswritebarrierrec 则反之。go:nowritebarrierrec 用于防止写屏障实现中的无限循环。

两个标志都在调度器中使用。写屏障需要一个活跃的 P (getg().m.p != nil)且调度器代码通常在没有活跃 P 的情况下运行。 在这种情况下,go:nowritebarrierrec 用于释放 P 的函数上,或者可以在没有 P 的情况下运行。 而且go:nowritebarrierrec 还被用于当代码重新要求一个活跃的 P 时。 由于这些都是函数级标注,因此释放或获取 P 的代码可能需要分为两个函数。

这两个指令都在调度程序中使用。 写屏障需要一个活跃的P( getg().mp != nil)并且调度程序代码通常在没有活动 P 的情况下运行。 在这种情况下,go:nowritebarrierrec 用于释放P的函数或者可以在没有P的情况下运行并且去: 当代码重新获取活动P时使用 go:yeswritebarrierrec。 由于这些是功能级注释,因此释放或获取P的代码可能需要分为两个函数。

进一步阅读的参考文献

许可

Go under the hood | CC-BY-NC-ND 4.0 & MIT © changkun