参考 小林coding
Go的内存管理
Go使用的是TCMalloc算法,本质是分级分配和本地缓存。
分配架构
三级:mcache线程缓存、mcentral中央缓存、mheap页堆。
每个P有自己的mcache,避免锁竞争;
mcentral按照对象大小来分配;
mheap向操作系统申请大块内存。
内存逃逸
简单来说就是本来应该分配在栈的对象分配到了堆上,而根据栈和堆的特点,栈是在函数结束后能自动回收,而堆需要GC垃圾回收,导致浪费。
内存逃逸的情况
- 返回局部变量的指针 局部变量无法在函数退出时自动回收
- 动态扩容(map/slice) 无法在编译器确定大小
- interfce{} 因为需要运行时的类型信息
- 闭包引用外部变量 被闭包捕获的变量逃逸到堆上
内存泄漏
简单来说就是,无法被回收导致泄露。
出现的情况
- goroutine 比如等待写入channel或者等待从channel中接收,死循环等,无法正常退出的情况。
- channel 未使用的channel和等待的channel互相持有goroutine。
- map过大 map删除不会释放内存,而是标记删除(标记为空闲)不释放内存,等待GC回收。
- slice 一个小的slice引用大的slice,大的也不会被释放(因为底层是共享数组),需要拷贝出一个新的小slice。
解决方案
- goroutine 用context设置超时,保证goroutine有退出机制。
- slice 拷贝独立小slice副本。
- channel 用select+default,保证channel能结束。