内存管理
内存管理
xxx
内存分配
内存分配有两种方式:栈分配和堆分配- 栈分配是在函数调用时为局部变量分配内存,当函数返回时,这些内存会自动释放
- 堆分配则是通过 new 或者 make 函数动态分配内存,需要GC释放编译器会自动选择在栈上还是在堆上,来分配局部变量的存储空间
如果一个局部变量在函数返回后仍然被使用,这个变量会从heap,而不是stack中分配内存
内存逃逸
原本应该在栈上分配的内存被分配到了堆上
- 变量的生命周期超出了其作用域
- 大对象的分配
- 闭包引用
- 在 interface 类型上调用方法
- 在 slice 或 map 中存储指针
go build - gcflags - m main. go
- 严格限制变量的作用域
- 对于小型的数据,使用传值而不是传指针
- 避免使用长度不固定的slice切片
内存泄漏
- 在程序执行过程中,已不再使用的内存空间没有被及时释放或者释放时出现了错误
- 导致这些内存无法被使用,直到程序结束这些内存才被释放
- 暂时性内存泄漏- 获取长字符串中的一段导致长字符串未释放- 获取长slice中的一段导致长slice未释放- defer 导致的内存泄漏
- 永久性内存泄漏- goroutine 泄漏导致内存泄漏- 定时器使用不当,time. Ticker未关闭导致内存泄漏- 不正确地使用终结器(Finalizers)导致内存泄漏
goroutine泄漏导致内存泄漏
goroutine 阻塞,无法继续向下执行,导致此 goroutine 关联的内存都无法释放,进一步造成内存泄漏
1. goroutine本身的栈所占用的空间造成内存泄漏
2. goroutine中的变量所占用的堆内存导致堆内存泄漏
- channel 阻塞
- select 操作在所有case 上都阻塞,造成内存泄漏
- 互斥锁没有释放,互斥锁死锁
- 申请过多的goroutine来不及释放
- 在 goroutine 里调用的一些资源,没有 close
1. go pprof获取goroutine profile文件
2 . top、traces、list定位内存泄漏的原因
判断依据:在节点正常运行的情况下,隔一段时间获取goroutine的数量- 如果后面获取的那次,某些goroutine比前一次多,- 如果多获取几次,是持续增长的- 就极有可能是goroutine泄漏
不正当使用内存场景
大数组作为参数导致短期内内存激增 goroutine阻塞拥挤等待,浪费内存