panic会中断程序执行,但可通过defer中的recover捕获并恢复,避免崩溃;在HTTP中间件中使用recover可防止单个请求panic导致服务终止;recover仅在当前goroutine有效,需在每个可能panic的协程内单独设置;不应滥用recover,仅用于处理不可控的异常情况,如第三方库错误或插件执行,常规错误应使用error处理。

在Go语言中,panic 是程序遇到无法继续执行的错误时触发的机制。它会中断正常流程并开始堆栈展开,逐层退出函数调用。但通过 recover,可以在 defer 函数中捕获 panic,实现安全恢复,避免程序崩溃。
recover 只能在 defer 修饰的函数中生效。当 panic 发生时,被 defer 的函数会被执行,此时调用 recover 可以阻止 panic 继续向上蔓延。
基本模式如下:
func safeRun() { defer func() { if r := recover(); r != nil { log.Printf("recovered: %v", r) } }() panic("something went wrong") }在这个例子中,程序不会崩溃,而是打印错误信息后继续执行后续代码。
立即学习“go语言免费学习笔记(深入)”;
Web 服务常使用 recover 避免单个请求导致整个服务中断。典型做法是在中间件中封装 recover 逻辑。
示例:
func recoverMiddleware(next http.HandlerFunc) http.HandlerFunc { return func(w http.ResponseWriter, r *http.Request) { defer func() { if r := recover(); r != nil { log.Printf("panic: %v\n", r) http.Error(w, "Internal Server Error", 500) } }() next(w, r) } }这样即使某个处理函数 panic,也能返回 500 错误,而不是让服务器退出。
recover 不应作为常规错误处理手段。只在以下场景考虑使用:
对于可预期的错误,应使用 error 返回值处理。recover 应用于“异常”而非“错误”。
recover 只能恢复当前 goroutine 的 panic。如果在新 goroutine 中发生 panic,外层的 defer 无法捕获。
正确做法是每个可能 panic 的 goroutine 内部都设置 recover:
go func() { defer func() { if r := recover(); r != nil { log.Println("goroutine panic recovered:", r) } }() // 可能 panic 的操作 }()否则,未捕获的 panic 会导致整个程序退出。
基本上就这些。合理使用 recover 能提升程序健壮性,但关键在于明确边界:只在必要时恢复,且恢复后应记录上下文以便排查问题。
以上就是Golang panic发生时如何安全恢复的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号