Go中goroutine的panic不会自动传播,必须在该goroutine内用defer包裹recover才能捕获;每个关键goroutine需独立处理panic,recover后应记录日志并依业务决策重启或降级,避免defer中再panic。
在 Go 中,协程(goroutine)里发生的 panic 不会自动传播到主 goroutine,若不显式 recover,会导致该 goroutine 悄悄终止,还可能引发资源泄漏或逻辑中断。要真正保障程序稳定,不能只靠 defer + recover,而需结合场景合理使用。
recover 只有在 defer 函数中、且 panic 正在被抛出时调用才有意义。如果写在普通代码流里,或者 recover 调用位置不在 panic 的同一 goroutine 中,它将返回 nil,起不到作用。
Go 不支持跨 goroutine 捕获 panic,所以每个可能出错的并发任务都得自己负责错误兜底。常见做法是在启动 goroutine 时就封装好 recover 逻辑。
例如:
go func() {
defer func() {
if r := recover(); r != nil {
log.Printf("goroutine panicked: %v", r)
// 可选:上报监控、清理资源、重试等
}
}()
// 业务逻辑,可能 panic
riskyOperation()
}()
recover 不是“吞掉错误”,而是获得一次干预机会。简单打印日志往往不够,还需结合业务判断后续动作:
recover 本身不解决根本问题,若 defer 里又发生 panic(比如日志写入失败、锁冲突),会导致原 panic 丢失,调试更困难。