Go中goroutine的panic不会跨协程传播,必须在每个可能panic的goroutine内用defer+recover捕获处理;recover仅在同一goroutine内有效,需注意状态一致性与资源清理。
在 Go 中,协程(goroutine)中发生的 panic 不会自动传播到启动它的 goroutine,也不会终止整个程序,但若未显式 recover,该 goroutine 会静默退出,可能导致资源泄漏、逻辑中断或难以排查的问题。因此,必须在可能 panic 的 goroutine 内部使用 recover 进行捕获和处理。
Go 的 panic/recover 机制仅在**同一 goroutine 内有效**。主 goroutine 中的 defer + recover 无法捕获子 goroutine 的 panic。所以每个可能出错的 goroutine 都应自行包裹 defer-recover 逻辑。
标准写法是用 defer 延迟一个包含 recover 的匿名函数,在 panic 发生后立即捕获并处理。
go func() {
defer func() {
if r := recover(); r != nil {
log.Printf("goroutine panic recovered: %v", r)
// 可选:上报错误、清理资源、重试等
}
}()
// 可能 panic 的业务代码
doRiskyWork()
}()
ver 只能捕获当前 goroutine 的 panic,不能“继续抛出”给其他 goroutine为避免重复写 defer-recover,可封装通用 recover 辅助函数,支持传入 context、logger 或回调处理。
立即学习“go语言免费学习笔记(深入)”;
func safeGo(f func(), logger *log.Logger) {
go func() {
defer func() {
if r := recover(); r != nil {
if logger != nil {
logger.Printf("panic in goroutine: %+v", r)
} else {
log.Printf("panic in goroutine: %+v", r)
}
// 可在此处调用 runtime/debug.PrintStack() 输出堆栈
}
}()
f()
}()
}
// 使用示例
safeGo(func() {
panic("something went wrong")
}, log.Default())
recover 并不等于“错误已解决”。它只是阻止了 goroutine 终止,但程序状态可能已损坏。
runtime/debug.Stack() 获取 panic 时的调用栈,辅助定位问题