当前位置: 首页> 教育> 高考 > 网站设计三把火科技_浙江关键词优化_杭州网站优化多少钱_厦门网页搜索排名提升

网站设计三把火科技_浙江关键词优化_杭州网站优化多少钱_厦门网页搜索排名提升

时间:2025/7/13 15:08:33来源:https://blog.csdn.net/m0_74282926/article/details/147600585 浏览次数:0次
网站设计三把火科技_浙江关键词优化_杭州网站优化多少钱_厦门网页搜索排名提升

使用 defer 和 recover 捕获 panic 的示例

在 Go 语言中,deferrecover 是处理 panic 的两个重要机制。下面是一个完整的示例,展示如何使用 defer 语句中的 recover 来捕获并处理 panic:

package mainimport ("fmt"
)func main() {// 正常执行fmt.Println("调用safeDivide(10, 2)")safeDivide(10, 2)// 触发 panic 的情况fmt.Println("\n调用safeDivide(10, 0)")safeDivide(10, 0)// 程序继续执行fmt.Println("\n程序继续执行...")
}func safeDivide(a, b int) {// 使用 defer 定义一个匿名函数来处理可能的 panicdefer func() {if r := recover(); r != nil {fmt.Printf("捕获到 panic: %v\n", r)fmt.Println("已恢复执行")}}()fmt.Printf("计算 %d / %d\n", a, b)result := divide(a, b)fmt.Printf("结果是: %d\n", result)
}func divide(a, b int) int {if b == 0 {// 当除数为0时,主动触发 panicpanic("除数不能为零!")}return a / b
}

输出结果:

调用safeDivide(10, 2)
计算 10 / 2
结果是: 5调用safeDivide(10, 0)
计算 10 / 0
捕获到 panic: 除数不能为零!
已恢复执行程序继续执行...

关键点说明:

  1. defer 语句确保函数退出时执行恢复逻辑
  2. recover() 只在 defer 函数中有效,它会捕获 panic 并返回 panic 的值
  3. 如果没有发生 panic,recover() 返回 nil
  4. 使用这种模式可以防止 panic 导致整个程序崩溃
  5. 恢复后程序会从 panic 点之后继续执行(实际上是继续执行 defer 之后的代码)

实际应用场景

这种技术常用于:

  • 防止第三方库的 panic 导致程序崩溃
  • 在关键服务中实现优雅的错误恢复
  • 在测试中捕获预期中的 panic

注意:不应该滥用 panic/recover 作为常规错误处理机制,Go 中通常使用 error 返回值来处理可预期的错误。


案例

1. 关键服务中实现优雅的错误恢复

package mainimport ("fmt""log""net/http"
)// recoverMiddleware 是一个中间件,用来捕获处理过程中所有的 panic
func recoverMiddleware(next http.Handler) http.Handler {return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {defer func() {if err := recover(); err != nil {// 记录错误日志,避免服务崩溃log.Printf("recover from panic: %v, request: %s %s", err, r.Method, r.URL.Path)// 返回 500 错误响应给客户端,防止崩溃信息泄露http.Error(w, "Internal Server Error", http.StatusInternalServerError)}}()// 调用下一个处理器next.ServeHTTP(w, r)})
}// simulateThirdPartyLibrary 是模拟的第三方库,可能会 panic
func simulateThirdPartyLibrary() {panic("第三方库炸了: connection pool corrupted")
}// handleRequest 是业务处理函数,调用了可能 panic 的第三方库
func handleRequest(w http.ResponseWriter, r *http.Request) {fmt.Fprintln(w, "业务处理中...")simulateThirdPartyLibrary() // 调用第三方库fmt.Fprintln(w, "业务处理完成") // 不会执行到这里
}func main() {// 注册路由,包上 recover 中间件http.Handle("/", recoverMiddleware(http.HandlerFunc(handleRequest)))log.Println("服务启动在 :8080")if err := http.ListenAndServe(":8080", nil); err != nil {log.Fatalf("服务启动失败: %v", err)}
}
  • recoverMiddleware 是标准写法:统一捕获每一次 HTTP 请求里的 panic,不影响其他请求继续工作。

  • 如果 simulateThirdPartyLibrary() 发生了 panic,recover 能记录日志并且返回安全的 500 错误,而不会让服务器整体挂掉。

  • 防止了单次异常影响整个服务,特别适合防御那些质量差但必须依赖的第三方库。


2. 防止第三方库 panic 导致程序崩溃

package mainimport ("encoding/json""fmt""log"
)// 第三方服务客户端
type ThirdPartyClient struct{}func (c *ThirdPartyClient) ParseResponse(data []byte) (map[string]interface{}, error) {defer func() {if r := recover(); r != nil {log.Printf("第三方库发生 panic: %v", r)}}()// 模拟第三方库内部可能 panic 的情况var result map[string]interface{}if err := json.Unmarshal(data, &result); err != nil {return nil, fmt.Errorf("解析失败: %w", err)}// 模拟第三方库可能 panic 的逻辑if _, ok := result["critical"]; !ok {panic("缺少 critical 字段") // 第三方库的不合理设计}return result, nil
}func main() {client := &ThirdPartyClient{}// 测试正常情况goodData := []byte(`{"critical": true, "value": "正常数据"}`)if res, err := client.ParseResponse(goodData); err != nil {log.Printf("处理正常数据时出错: %v", err)} else {log.Printf("正常数据结果: %v", res)}// 测试会触发第三方库 panic 的情况badData := []byte(`{"value": "缺少critical字段"}`)if res, err := client.ParseResponse(badData); err != nil {log.Printf("处理异常数据时出错: %v", err)} else {log.Printf("异常数据结果: %v", res)}// 程序继续执行log.Println("主程序继续运行...")
}

https://github.com/0voice

关键字:网站设计三把火科技_浙江关键词优化_杭州网站优化多少钱_厦门网页搜索排名提升

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

责任编辑: