【Go专家编程——内存管理——逃逸分析】

逃逸分析

逃逸分析(Escape Analysis)是指由编译器决定内存分配的位置,不需要程序员决定。

  • 在函数中申请一个新的对象
    • 如果分配在栈上,则函数执行结束后可自动将内存回收
    • 如果分配在堆上,则函数执行结束后可交给GC(垃圾回收)处理
  • 有了逃逸分析,返回函数局部变量变得可能
  • 逃逸分析还跟闭包息息相关

1.逃逸策略

在函数中申请新对象时,编译器会根据该对象是否被函数外部引用来决定是否逃逸:

  • 如果函数外部没有引用,则优先放到栈中
  • 如果函数外部存在引用,则必定放到堆中
  • 注意点:对于仅在函数内部使用的变量,也有可能放入堆中,如内存过大超过栈的存储能力。

2. 逃逸场景

我们可以通过编译参数-gcflags=-m来查看编译过程中的逃逸分析过程

例如go build -gcflags=-m

2.1 指针逃逸

go 复制代码
type Student struct {
	Name string
	Age int
}

func StudentRegister(name string,age int) *Student{
	s := new(Student)	// 局部变量s逃逸到堆中
	s.Name = name
	s.Age = age
	return s
}
func main(){
	StudentRegister("Jim", 18)
}

函数StudentRegister()内部的s为局部变量,其值通过函数返回值返回,s本身为一个指针,其指向的内存地址不会是栈而是堆。

2.2 栈空间不足逃逸

go 复制代码
func Slice(){
	s := make([]int,1000,1000)
	for index,_ := range s{
		s[index] = index
	}
}
func main(){
	Slice()
}

Slice函数分配了一个长度为1000的切片,是否逃逸取决于栈的空间是否足够大。

当切片长度不断增加到10000时就会发生逃逸

  • 实际上当栈的空间不足以存放当前对象或无法判断当前切片长度时会将对象分配到堆中

2.3 动态类型逃逸

很多函数的参数为interface类型,比如fmt.Println(a ...interface{}),编译期间很难确定其参数具体类型,也会产生逃逸。

go 复制代码
func main(){
	s := "Escape"
	fmt.Println(s) 
}

2.4 闭包引用对象逃逸

go 复制代码
func Fibonacci() func() int {
	a,b := 0,1
	return func() int {
		a, b = b,a+b
		return a
	}
}

func main(){
	f := Fibonacci()
	for i:=0;i<10;i++{
		fmt.Printf("Fibonacci: %d\n",f())
	}
}

该函数返回一个闭包,闭包使用了函数的局部变量a,b,使用时通过该函数获取闭包,然后每次执行闭包都会一次输出Fibonacci数列。

Fibonacci()函数中原本属于局部变量的a和b由于闭包的引用,不得不将两者放到堆上,以至于产生了逃逸。

3. 小结

  • 栈上分配内存比在堆上分配内存有更高的效率
  • 栈上分配的内存不需要GC处理
  • 堆上分配的内存使用完毕会交给GC处理
  • 逃逸分析的目的是决定分配地址是栈还是堆
  • 逃逸分析在编译阶段完成
相关推荐
人间打气筒(Ada)5 小时前
「码动四季·开源同行」go语言:微服务网关如何作为服务端统一入口点?
微服务·golang·开源·微服务网关·go实战
asyxchenchong8885 小时前
农业系统模拟APSIM全流程详解(气象/土壤/碳氮平衡/NG版本)附R批量处理代码
开发语言·r语言
IT_陈寒5 小时前
Java线程池用完不关闭?小心内存泄漏找上门
前端·人工智能·后端
跟着珅聪学java5 小时前
在 Java 中处理 JSON 去除空 children数组,可以使用 Jackson 库。这里有几种实现方式
开发语言·windows·python
计算机安禾5 小时前
【数据结构与算法】第33篇:交换排序(二):快速排序
c语言·开发语言·数据结构·数据库·算法·矩阵·排序算法
William Dawson5 小时前
Java 后端高频 20 题超详细解析 ①
java·开发语言
lly2024065 小时前
PHP 魔术常量
开发语言
Evand J5 小时前
【MATLAB例程分享】三维非线性目标跟踪,观测为:距离+方位角+俯仰角,使用无迹卡尔曼滤波(UKF)与RTS平滑,高精度定位
开发语言·matlab·目标跟踪
编程之升级打怪5 小时前
Java NIO的简单封装
java·开发语言·nio
小江的记录本5 小时前
【JEECG Boot】 《JEECG Boot 数据字典使用教程》(完整版)
java·前端·数据库·spring boot·后端·spring·mybatis