简介

每个版本的Golang的垃圾回收都在不断优化中,而且方法和策略都在变化,因此这里只是总结出以下几个关键点:

  • 什么样的数据需要GC
  • 触发GC的条件是什么
  • GC时发生了什么
  • 能否从代码层面上提高GC的效率

GC的基本流程

Golang在确定的时间,或者内存分配到达一定程度时,进行GC。GC时,会停止STW(Stop The World),即对外的服务都会暂停,然后进行垃圾回收处理。Go1.12引入了三色标记法和write-barrier的方式;在Go1.14中,引入看了抢占式回收机制。

write-barrier机制:
假设有4个G在运行,如下图:

进行GC的时候,需要STW,此时的4个G都要停止工作。如果有一个没有停止工作的,则GC暂时不能发生。比如下图:

第4个G没停止工作,则GC需要等待其结束。Go1.14中,可以抢占第4个G的工作状态,保存其状态后,再进行GC

GC的时候,GC机制会征用一些G并发辅助进行工作,一般有25%的G会被征用。

整体工作流程

  • 创建白、灰和黑三个集合
  • 初始化所有的待回收对象都是白色的
  • 从根节点遍历对象,不递归;遍历到的白色对象放到灰色集合当中
  • 之后遍历灰色集合,把灰色对象引用的对象,从白色集合中放入灰色集合,并把现在的灰色对象放入黑色集合中
  • 重复上一步,知道灰色集合是空的
  • 通过write-barrier检测对象的变化,重复以上操作
  • 回收所有的白色对象

GC回收的对象

永远不要过早的优化程序!!!

栈内存分配和回收的代价远远小于堆内存。Golang的垃圾回收发生在全局的堆上和每个Goroutinue的栈上。回收栈内存只需要两个CPU指令,push和pop。然而,分配在栈内存的数据,需要在编译期间就得知道type和size。

Golang的编译器使用“逃逸分析”的方式,决定采取栈回收还是堆回收的方式。如果发生逃逸,则使用堆回收。
go build -gcflags '-m'命令可以分析内存逃逸的现象。

发生内存逃逸的几种情况:

  • chan中发送数据的指针或者包含指针的值。
    因为编译器此时不知道值什么时候会被接收,因此只能放入堆中
  • 非直接的函数调用,比如在闭包中引用包外的值,因为闭包执行的生命周期可能会超过函数周期,因此需要放入堆中
  • 在slice或map中存储指针或者包含指针的值。
    比如[]*string,即使slice是在栈上,其存储的值仍然会放入堆中
  • slice如果底层使用array作为容器,在使用append扩容的时候。但是,如果知道具体扩容的数量,则仍然在栈上。
    如果在编译期间,slice知道自己的size,那么放入栈中。更多的时候,是不知道size的,比如append的时候,此时只能放入堆中。
  • interface类型多态的时候调用方法,此时会发生逃逸
  • 指针、slice和map作为参数返回的时候,此时肯定要发生逃逸。

总结一下发生逃逸的结论:

  1. 首先明确一点,Golang中所有的数据都是按值传递,这点和C语言是一样的(注意Golang中的数组名是值,和C的差别)。所谓的map、slice和chan等是引用,其本质原因是,这些结构的内部都有指针,复制的时候,内部都是复制的指针,因此表现的是传值。
  2. 在函数调用中,对于指针的情况,只要指向的地址的所有者只有一个,那么必然是栈回收;而一旦存在地址存在不确定变化时,则转换成堆的数据。比如slice情况,因为slice会扩容或者缩容,因此造成不确定情况。

以下使用代码示例说明:

package main
func main() {ch := make(chan int, 1)x := 5ch <- x  // x不发生逃逸,因为只是复制的值ch1 := make(chan *int, 1)y := 5py := &ych1 <- py  // y逃逸,因为地址传入了chanz := 5pz := &z  // z不逃逸,因为是确定性析构*pz += 1}

执行命令:go build -gcflags ./main.go,得到结论是y z都发生了逃逸。

# command-line-arguments
.\main.go:3:6: can inline main as: func() { ch := make(chan int, 1); x := 5; ch <- x; ch1 := make(chan *int, 1); y := 5; py := &y; ch1 <- py; z := 5; pz := &z; *pz += 1 }
.\main.go:9:2: y escapes to heap:
.\main.go:9:2:   flow: py = &y:
.\main.go:9:2:     from &y (address-of) at .\main.go:10:8
.\main.go:9:2:     from py := &y (assign) at .\main.go:10:5
.\main.go:9:2:   flow: {heap} = py:
.\main.go:9:2:     from ch1 <- py (send) at .\main.go:11:6
.\main.go:9:2: moved to heap: y

如果使用slice和map的模式:

package main
func main() {var x intx = 10var ls []*intls = append(ls, &x)var y intvar mp map[string]*intmp["y"] = &y
}

结论分析:

# command-line-arguments
.\main.go:3:6: can inline main as: func() { var x int; x = <N>; x = 10; var ls []*int; ls = <N>; ls = append(ls, &x); var y int; y = <N>; var mp map[string]*int; mp = <N>; mp["y"] = &y }
.\main.go:4:6: x escapes to heap:
.\main.go:4:6:   flow: {heap} = &x:
.\main.go:4:6:     from &x (address-of) at .\main.go:7:18
.\main.go:4:6:     from append(ls, &x) (call parameter) at .\main.go:7:13
.\main.go:9:6: y escapes to heap:
.\main.go:9:6:   flow: {heap} = &y:
.\main.go:9:6:     from &y (address-of) at .\main.go:11:12
.\main.go:9:6:     from mp["y"] = &y (assign) at .\main.go:11:10
.\main.go:4:6: moved to heap: x
.\main.go:9:6: moved to heap: y

使用闭包捕获指针的模式:

package main
import "time"
func main() {x := 10go func(x *int) {*x += 1}(&x) // 捕获的瞬间,x没有移动到heap上,但是整个闭包移动到了heap上,因此x也跟随闭包被移动到heap上了time.Sleep(time.Second * 2)
}

结论分析:

# command-line-arguments
.\main.go:5:6: cannot inline main: unhandled op GO
.\main.go:7:5: can inline main.func1 as: func(*int) { *x += 1 }
.\main.go:7:5: func literal escapes to heap:
.\main.go:7:5:   flow: {heap} = &{storage for func literal}:
.\main.go:7:5:     from func literal (spill) at .\main.go:7:5
.\main.go:7:5:     from go (func literal)(&x) (call parameter) at .\main.go:7:2
.\main.go:6:2: x escapes to heap:
.\main.go:6:2:   flow: {heap} = &x:
.\main.go:6:2:     from &x (address-of) at .\main.go:9:4
.\main.go:6:2:     from go (func literal)(&x) (call parameter) at .\main.go:7:2
.\main.go:7:10: x does not escape
.\main.go:6:2: moved to heap: x
.\main.go:7:5: func literal escapes to heap

对于slice扩容的情况:

package main
import ("os""strconv"
)
func main() {ls := []int{1, 2, 3}ls = append(ls, 4) // 确定性的,不逃逸,编译期间可以知道var n intn, _ = strconv.Atoi(os.Args[1])  // 输入数据后,则结果不可知,因此可能逃逸ls1 := []int{1, 2, 3}for i := 0; i < n; i++ {ls1 = append(ls1, 1)}
}

interface类型的GC,涉及使用interface类型转换并调用对应的方法的时候,都会发生内存逃逸,给出代码示例:

package main
type foo interface {fooFunc()
}
type foo1 struct{}
func (f1 foo1) fooFunc() {}
type foo2 struct{}
func (f2 *foo2) fooFunc() {}
func main() {var f foof = foo1{}f.fooFunc()   // 调用方法时,发生逃逸,因为方法是动态分配的f = &foo2{}f.fooFunc()
}

执行说明:

go-code ➤ go build -gcflags "-m" main.go
# command-line-arguments
.\main.go:9:6: can inline foo1.fooFunc
.\main.go:13:6: can inline (*foo2).fooFunc
.\main.go:13:7: f2 does not escape
.\main.go:17:4: foo1 literal escapes to heap
.\main.go:19:6: &foo2 literal escapes to heap
<autogenerated>:1: leaking param: .this
<autogenerated>:1: inlining call to foo1.fooFunc
<autogenerated>:1: .this does not escape

返回slice等的情况:

package mainfunc foo() []int {return []int{1, 2, 3}
}func main() {ls := foo() // 发生逃逸ls = append(ls, 1)
}

分析结果:

# command-line-arguments
.\main.go:3:6: can inline foo as: func() []int { return []int literal }
.\main.go:7:6: can inline main as: func() { ls := foo(); ls = append(ls, 1) }
.\main.go:8:11: inlining call to foo func() []int { return []int literal }
.\main.go:4:14: []int literal escapes to heap:
.\main.go:4:14:   flow: ~r0 = &{storage for []int literal}:
.\main.go:4:14:     from []int literal (spill) at .\main.go:4:14
.\main.go:4:14:     from return []int literal (return) at .\main.go:4:2
.\main.go:4:14: []int literal escapes to heap
.\main.go:8:11: []int literal does not escape

传值还是传指针的问题
根据上面的分析,指针更容易出现内存逃逸的现象。而一旦发生了内存逃逸,则不可避免地对GC造成潜在的压力。有种错误的观念:传指针的代价总是比传值的拷贝代价小。这种观念只在像C语言这种没有GC的低级语言中可能适用。原因如下:

  • 对指针解引用的时候,编译器会进行一些检查。
  • 指针一般都不是临近地址的引用,而复制时,一般都是CPU cash中的数据,cash line内的数据的复制,速度基本和一个复制指针相等

因此,对于小型的数据,一般传值就够了。在某些情况下,需要对代码做一些重构,以消除成员变量中不必要的指针类型。slice有些情况下,可能也会造成内存逃逸,使用已知固定长度的slice,某些情况下会减少内存逃逸。nterface调用方法会发生内存逃逸,某些热点的情况下,可以考虑优化interface的情况。

几个总结

  • 永远不要过早的优化,使用数据驱动优化代码
  • 栈回收的代价远远小于堆回收的代价
  • 指针一般情况下会影响栈回收
  • 在热点代码片段,谨慎的使用interface

参考文档

  • https://juejin.im/post/5d2825bff265da1b6836e8d4
  • https://spin.atomicobject.com/2014/09/03/visualizing-garbage-collection-algorithms/
  • https://segment.com/blog/allocation-efficiency-in-high-performance-go-services/
  • https://i6448038.github.io/2019/03/04/golang-garbage-collector/
  • http://idiotsky.top/2017/08/16/gc-three-color/

Golang的GC和内存逃逸相关推荐

  1. golang byte转string_golang面试题:怎么避免内存逃逸?

    问题 怎么避免内存逃逸? 怎么答 在runtime/stubs.go:133有个函数叫noescape.noescape可以在逃逸分析中隐藏一个指针.让这个指针在逃逸分析中不会被检测为逃逸. // n ...

  2. golang int 转string_Golang的逃逸分析

    逃逸分析 逃逸分析(Escape Analysis)指的是将变量的内存分配在合适的地方(堆或者栈). 在函数中申请内存有2种情况: - 如果内存分配在栈(stack)上,当函数退出的时候,这部分内存会 ...

  3. go java gc_图解Golang的GC垃圾回收算法

    虽然Golang的GC自打一开始,就被人所诟病,但是经过这么多年的发展,Golang的GC已经改善了非常多,变得非常优秀了. 以下是Golang GC算法的里程碑: v1.1 STW v1.3 Mar ...

  4. 图解Golang的GC算法

    图解Golang的GC算法 原创 RyuGou 程序猿菜刚RyuGou 2019-03-10 来源:https://mp.weixin.qq.com/s/_h0-8hma5y_FHKBeFuOOyw ...

  5. JVM详解(类加载,内存分配,GC,内存模型)

    一. 背景 1. 前言 作为一位Java开发工作者,在关心业务代码开发的同时,我们也需要了解java底层时如何运作的,了解为什么说java是跨平台的语言,所以这一篇对JVM(java虚拟机)进行剖析和 ...

  6. GO语言内存逃逸图文分析

    在分析内存逃逸之前我们需要知道我们的内存空间的分布图,为内存逃逸的分析做一些铺垫,这是内存分布,当系统知道变量什么时候的生命周期和内存分配大小的时候我们都将其分配在栈区 (必知)分析内存逃逸之前我们需 ...

  7. 关于Java与Golang的GC

    关于Java与Golang的GC nanko 文章目录 一.GC的普遍解决方案 1.标记 1.1.引用计数法 1.2.可达性分析 1.3 二者对比 2.清理 2.1 mark-sweep 2.2 co ...

  8. java golang gc_Golang GC 垃圾回收机制详解

    摘要 在实际使用 go 语言的过程中,碰到了一些看似奇怪的内存占用现象,于是决定对go语言的垃圾回收模型进行一些研究.本文对研究的结果进行一下总结. 什么是垃圾回收? 曾几何时,内存管理是程序员开发应 ...

  9. C++简单实现GC和内存池

    C++简单实现GC和内存池 本章前言: 这章就是为了避开传统的new\delete创建堆对象的手法,如果使用不当指针容易出现问题,所以本章简单的实现了引用计数垃圾回收,使用起来貌似还不错.Bug还有待 ...

最新文章

  1. 使用 json_serializable (flutter packages pub run build_runner build) 问题
  2. jpa 根据主键生成策略获取id_JPA主键生成策略
  3. ajax请求解析json,如何为Ajax请求解析json响应?
  4. 微信浏览器中IOS12键盘回弹的问题
  5. 知道cve编号怎么搜poc_想在海南买房不知道该怎么选?快来看看这些热搜楼盘吧!...
  6. 使用ASP.NET MVC构建HTML5离线web应用程序
  7. php 中echo命令详解,DOS中的ECHO命令详解
  8. HTC Z710t解锁 获取root权限
  9. vivado基本知识
  10. 证件照素材大合集(全网最全版本,免费分享)
  11. 麦克林托克和转座基因
  12. 上传图片报413错误
  13. Delta, Hudi, Iceberg对比
  14. 【超详细】Docker从入门到干活,就看这一篇文章
  15. 唯品会狂飙突进的去向
  16. 百钱百鸡php穷举法,PHP经典题:百钱百鸡问题(穷举算法)
  17. gist feature
  18. 工业dcs系统有服务器吗,DCS是什么 你了解多少
  19. 重载与重写、重用与重构
  20. 【笔记】29元microbit套装如何玩——套装硬件简介

热门文章

  1. python3.7模块内容_python3.7 time模块
  2. Being a good boy in the spring festival NIM博弈
  3. 【文献阅读】Self-Normalizing Neural Networks
  4. 最长回文(Manacher算法模板)
  5. nlp-tutorial代码注释3-1,RNN简介
  6. Java 1.2.1 读取输入
  7. 深度学习——loss函数的学习笔记(legacy)
  8. 【java】计算 list 中每一行的合计值
  9. 缓存淘汰策略——最近最久未使用策略(LRU)
  10. 关于Oracle创建新的监听和服务名的问题一二