浅谈Golang逃逸分析

什么是内存逃逸

通常情况下,函数里的局部变量 会被分配到 内存空间上,当函数执行结束,这些局部变量所占用的内存会被自动释放。然而,若变量的生命周期超出了函数的作用范围,编译器会将该变量分配到堆内存上,这种现象就叫做内存逃逸。

什么情况下会发生内存逃逸

逃逸分析是一种静态代码分析技术,Go 编译器在编译时会对代码进行数据流和控制流分析,以确定变量的生命周期和作用域,其核心目标是判断变量是否会在函数外部被引用,如果会,则该变量需要被分配到堆上,发生内存逃逸;如果不会,则可以安全地分配到栈上。

Golang程序变量会携带有一组校验数据,用来证明它的整个生命周期和作用域是否在运行时完全可知。如果变量通过了这些校验,它就可以在栈上分配。否则就说它 逃逸 了,必须在堆上分配

可以使用该命令辅助分析,go build -gcflags="-m -l" ***.go

典型的会引起内存逃逸的情况:

  1. 函数将局部变量的指针返回:局部变量原本应该在栈中分配,在栈中回收。因为函数执行结束后,栈上的局部变量会被销毁,如果返回其指针,外部代码就无法正常访问该变量,所以编译器会将其分配到堆上。
go 复制代码
func test() *int {
    num := 10
    return &num
}

func main() {
    fmt.Println(test())
}
  1. 发送指针变量或包含指针的值到channel中:像结构体、切片、映射等数据类型,若其内部包含指针类型的字段;或者直接将一个变量的指针发送到channel中,在编译时,没有办法知道哪个 goroutine 会接收该数据。无法保证接收方在函数结束后还会不会通过该指针访问变量,所以编译器会把变量分配到堆上。
go 复制代码
func test2(ch chan *int) {
    a := 1
    ch <- &a
}

func main() {
    ch := make(chan *int)
    go test2(ch)
    <-ch
}

3. 切片发生扩容,并且作为返回值或闭包使用 :其实就算不发生扩容,把一个切片作为函数返回值,或者闭包使用,也会发生逃逸,只不过强调扩容是因为会使切片的内存管理更加复杂,内存逃逸的现象更加明显,对程序的性能和内存使用可能产生更大的影响,需要开发者更加关注和谨慎处理。

  1. 在一个切片上存储指针或带指针的值。
go 复制代码
func main() {
    s := make([]*string, 0)
    c := "A"
    s = append(s, &c)

    o := Order{
       OrderID:   123,
       OrderInfo: &OrderInfo{},
    }
    orders := make([]Order, 0)
    orders = append(orders, o)
}
  1. 变量被 interface{} 类型引用 :如果将一个变量赋值给 interface{} 类型的变量,该变量可能会发生内存逃逸。因为 interface{} 类型可以存储任意类型的值,编译器无法确定其具体大小和生命周期,为了安全起见,会将其分配到堆上。
  2. 在 interface 类型上调用方法:在 interface 类型上调用方法都是动态调度的 ------ 方法的真正实现只能在运行时知道。想像一个 io.Reader 类型的变量 r , 调用 r.Read(b) 会使得 r 的值和切片b 的背后存储都逃逸掉,所以会在堆上分配。

如何避免内存逃逸

除了注意上面几种编码方式,尽量避免发生内存逃逸。

runtime/stubs.go:133有个函数叫noescapenoescape可以在逃逸分析中隐藏一个指针 。让这个指针在逃逸分析中不会被检测为逃逸。不过要注意,这个函数只是欺骗了逃逸分析器,实际上指针的引用关系依然存在。

go 复制代码
func test() *int {
    num := 10
    return &num
}

func testNotEscape() *int {
    num := 10
    // 使用 noescape 隐藏指针的引用关系
    ptr := (*int)(noescape(unsafe.Pointer(&num)))
    return ptr
}

func noescape(numPtr unsafe.Pointer) unsafe.Pointer {
    x := uintptr(numPtr)
    return unsafe.Pointer(x ^ 0)
}

func main() {
    test()
    testNotEscape()
}
相关推荐
huohuopro15 分钟前
thinkphp模板文件缺失没有报错/thinkphp无法正常访问控制器
后端·thinkphp
什码情况41 分钟前
星际篮球争霸赛/MVP争夺战 - 华为OD机试真题(A卷、Java题解)
java·数据结构·算法·华为od·面试·机试
chenyuhao20242 小时前
链表面试题7之相交链表
数据结构·算法·链表·面试·c#
陌尘(MoCheeen)3 小时前
技术书籍推荐(002)
java·javascript·c++·python·go
牛马baby3 小时前
Java高频面试之并发编程-16
java·开发语言·面试
cainiao0806053 小时前
《Spring Boot 4.0新特性深度解析》
java·spring boot·后端
-曾牛3 小时前
Spring AI 与 Hugging Face 深度集成:打造高效文本生成应用
java·人工智能·后端·spring·搜索引擎·springai·deepseek
Frankabcdefgh4 小时前
前端进化论·JavaScript 篇 · 数据类型
javascript·安全·面试·数据类型·操作符·初学者·原理解析
南玖yy4 小时前
C/C++ 内存管理深度解析:从内存分布到实践应用(malloc和new,free和delete的对比与使用,定位 new )
c语言·开发语言·c++·笔记·后端·游戏引擎·课程设计
计算机学姐4 小时前
基于SpringBoot的小区停车位管理系统
java·vue.js·spring boot·后端·mysql·spring·maven