sync.Pool可复用临时对象以减少堆分配和GC压力,适用于短生命周期、结构稳定且可重置的对象;需包级声明、成对调用Get/Put并安全重置,避免用于大对象、复杂状态或长生命周期场景。
Go 语言中,频繁创建和销毁小对象会触发大量堆分配,增加 GC 压力、降低性能。使用 sync.Pool 可以有效复用临时对象,减少内存分配次数和 GC 频率,尤其适合生命周期短、结构稳定、可重置的对象(如 buffer、request context、DTO 结构体等)。
sync.Pool 是一个并发安全的对象缓存池,它不保证对象一定被复用,也不保证对象存活时间。它的设计目标是“减缓分配压力”,而非“绝对复用”。关键特性包括:
Get() 时优先从本地池取,无则尝试共享池,最后调用 New 创建新对象Put() 时对象被放回本地池(若未满),否则丢弃或归入共享池推荐将 sync.Pool 定义为包级变量,并通过 New 字段提供对象构造逻辑。避免在函数内反复 new Pool 实例,也避免在 New 中返回 nil 或带副作用的对象。
例如,复用字节切片缓冲区:
var bufferPool = sync.Pool{
New: func() interface{} {
return make([]byte, 0, 1024) // 预分配容量,避免 append 时扩容
},
}
再如,复用自定义结构体(需确保可安全重置):
type RequestData struct {
ID int
Body []byte
Header map[string]string
}
var requestDataPool = sync.Pool{
New: func() interface{} {
return &RequestData{
Header: make(map[string]string),
}
},
}
必须成对使用:每次 Get() 后,在作用域结束前调用 Put() 归还对象(除非明确不再需要)。注意不要归还已逃逸到 goroutine 外部、或正被其他 goroutine 使用的对象。
常见安全写法(配合 defer):
buf := bufferPool.Get().([]byte)
defer bufferPool.Put(buf) // 确保归还
// 使用 buf...
buf = buf[:0] // 清空内容,保留底层数组
copy(buf, data)
// 若 buf 容量不足,可重新分配并显式放弃旧 buf(不 Put)
if len(data) > cap(buf) {
bufferPool.Put(buf)
buf = make([]byte, len(data))
}
对于结构体,务必在 Put 前重置字段(尤其是 map、slice、指针等):
req := requestDataPool.Get().(*RequestData)
defer func() {
req.ID = 0
req.Body = req.Body[:0]
for k := range req.Header {
delete(req.Header,
k)
}
requestDataPool.Put(req)
}()
不是所有场景都适合:
上线前建议用 go tool pprof 对比 heap profile 和 allocs profile,确认 sync.Pool 确实降低了 runtime.mallocgc 调用次数和堆分配总量。