协程泄漏本质是程序逻辑缺陷,因缺少退出机制或channel未关闭导致goroutine永久阻塞;应使用context.Context统一控制生命周期,通过ctx.Done()监听取消信号并及时清理。
协程泄漏不是Go语言的bug,而是程序逻辑缺陷:启动了goroutine,但因缺少退出机制或channel未关闭,导致其永远阻塞在select、recv或send上,无法被调度器回收。常见于后台监听、定时任务、管道处理等场景。
推荐使用context.Context作为协程的“退出信号源”,而非手动维护布尔标志或额外channel。父goroutine通过context.WithCancel或context.WithTimeout派生子context,子goroutine监听ctx.Done()通道,在收到信号后清理资源并返回。
ctx.Err() != nil,避免启动已取消的协程select中把作为必选项,优先响应取消
cancel()后,建议用sync.WaitGroup等待协程真正退出,确保资源释放完成channel只应由**发送方**关闭,且**只能关闭一次**;接收方关闭会panic;向已关闭channel发送数据也会panic。关闭时机必须明确——通常是所有发送操作结束之后,且不再有新数据要写入。
sync.Once包装关闭逻辑val, ok := 判断channel是否关闭(ok==false表示已关)
for range ch遍历channel,循环会在channel关闭后自动退出,但需确保发送方已关闭,否则会永久阻塞协程泄漏常源于“以为它结束了,其实卡住了”。比如启动一个goroutine读chann
el,但忘记关闭channel,或发送方提前退出没通知接收方;又如用无缓冲channel做同步,但某一方未执行收/发操作。
time.AfterFunc(5 * time.Minute, func(){ log.Println("goroutine stuck?") })
runtime.NumGoroutine()在测试或调试阶段观察协程数量趋势,发现异常增长ctx.Done()