Go - 内存逃逸

概念

每个函数都有自己的内存区域来存放自己的局部变量、返回地址等,这个内存区域在栈中进行分配。当函数结束时,这段内存区域会进行释放。

但有些变量,我们想在函数结束后仍然使用它,那么就要把这个变量在堆上分配,这种从"栈"上逃逸到"堆"上的现象就是内存逃逸。

在栈上分配的内存,由系统申请和释放,不会有额外的性能开销。

而在堆上分配的内存,如果要回收掉,就需要进行GC,内存逃逸额外带来的GC会导致性能开销变大。

逃逸机制

根据变量是否被外部引用来决定是否逃逸:

  1. 若函数外部没有引用,则优先放入栈中
  2. 若函数外部存在引用,则优先放入堆中
  3. 若栈上放不下,则必定放到堆中

逃逸分析

通过编译参数-gcflags=-m查看编译过程中的逃逸分析。

一般的逃逸类型:

指针逃逸

在函数中创建了一个对象,返回了这个对象的指针。此时,函数虽然推出了,但因为指针的存在,对象的内存不能随着函数结束而回收,只能逃逸到堆上。

go 复制代码
package main  
import "fmt"  
type Demo struct {  
	name string  
}  
func createDemo(name string) *Demo {  
	d := new(Demo) // 局部变量 d 逃逸到堆  
	d.name = name  
	return d  
}  
func main() {  
	demo := createDemo("demo")  
	fmt.Println(demo)  
}

interface{}动态类型逃逸

如果函数的参数为interface{}类型,编译期间很难确定其参数具体类型,也会发生逃逸。

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

demo作为实参传给Println(i interface{}) (n int, err error)方法,因为该函数的参数类型定义为interface{},因此会发生逃逸。

栈内存不足

操作系统对内核线程使用的栈空间是有大小限制的,因为栈空间通常较小,因此递归函数实现不当时,容易导致栈溢出。

对于Go语言来说,运行时(runtime)尝试在goroutine需要的时候动态地分配栈空间,goroutine的初始栈大小为2kb。

当goroutine被调度时,会绑定到内核线程执行,所以栈空间大小也不会炒股共操作系统的限制。

对于Go编译器来说,超过一定大小的局部变量将逃逸到堆上。

go 复制代码
// 1.
func generate8192() {  
	nums := make([]int, 8192) // = 64KB  
	for i := 0; i < 8192; i++ {  
		nums[i] = rand.Int()  
	}  
}  
// 2.
func generate8193() {  
	nums := make([]int, 8193) // > 64KB  
	for i := 0; i < 8193; i++ {  
		nums[i] = rand.Int()  
	}  
}  
// 3.
func generate(n int) {  
	nums := make([]int, n) // 不确定大小  
	for i := 0; i < n; i++ {  
		nums[i] = rand.Int()  
	}  
}  
  
func main() {  
	generate8192()  
    generate8193()  
    generate(1)  
}

2和3均逃逸到堆上,而1没有。

说明当切片内存超过一定大小,栈空间不足时,便会逃逸到堆上。

闭包

当闭包访问到其外层的函数作用域时,会发生内存逃逸。

go 复制代码
func Increase() func() int {  
	n := 0  
	return func() int {  
		n++  
		return n  
	}  
}  
  
func main() {  
	in := Increase()  
	fmt.Println(in()) 
}

Increase()返回值是一个闭包函数,该闭包函数引用了外部变量n,知道in()被销毁,n却不能随着函数退出而被回收,因此逃逸到堆上。

内存逃逸的影响

在栈上分配和回收内存的开销是很低的,只需要poppush命令,分别负责释放栈空间和分配数据内存。在栈上分配内存,消耗的仅是将数据拷贝到内存的时间。

而在堆上分配内存,很大的额外开销是垃圾回收。

如果频繁发生内存逃逸,会导致程序占用过多的内存资源,影响程序的性能和稳定性。主要体现在以下几个方面:

  • 内存占用增加:由于堆分配的内存不会自动释放,所以会导致程序占用的内存资源不断增加,特别是在长时间运行的程序中,可能会导致系统资源耗尽。
  • 性能下降:相比于栈分配,堆分配需要更多的 CPU 和内存资源,stw,因此会导致程序的运行速度变慢。
  • 程序不稳定:如果程序中存在大量的内存逃逸,可能会导致垃圾回收器频繁工作,从而影响程序的稳定性。
相关推荐
007php0075 小时前
某游戏互联网大厂Java面试深度解析:Java基础与性能优化(一)
java·数据库·面试·职场和发展·性能优化·golang·php
poemyang6 小时前
告别漫长GC停顿:深入解析G1如何实现可预测的毫秒级响应
java虚拟机·垃圾回收
豆浆whisky11 小时前
Go内存管理最佳实践:提升性能的Do‘s与Don‘ts|Go语言进阶(17)
开发语言·后端·golang
Yeats_Liao15 小时前
Go Web 编程快速入门 18 - 附录B:查询与扫描
开发语言·前端·后端·golang
小八四爱吃甜食15 小时前
【R语言】构建GO、KEGG相关不同物种的R包
开发语言·golang·r语言
赵文宇(温玉)1 天前
构建内网离线的“github.com“,完美解决内网Go开发依赖
开发语言·golang·github
草明1 天前
Go 的 IO 多路复用
开发语言·后端·golang
poemyang1 天前
吞吐量、延迟、内存:深入理解垃圾回收的“三元悖论”
java虚拟机·垃圾回收
绛洞花主敏明2 天前
Go切片的赋值
c++·算法·golang
007php0072 天前
某游戏大厂 Java 面试题深度解析(四)
java·开发语言·python·面试·职场和发展·golang·php