golang recover错误

可以recover的错误

  1. 显式触发的panic

    通过panic("error message")主动抛出的错误,只要在同一goroutinedefer链中调用recover,即可捕获并恢复。

    go 复制代码
    defer func() {
        if r := recover(); r != nil {
            fmt.Println("Recovered:", r)
        }
    }()
    panic("user-triggered panic")
  2. 运行时错误(部分)

    如切片越界、类型断言失败、空指针解引用等运行时panic,只要在defer中正确使用recover技术上可以捕获。但程序可能处于不可信状态,继续执行需谨慎。

    go 复制代码
    defer func() {
        if r := recover(); r != nil {
            fmt.Println("Recovered from runtime error:", r)
        }
    }()
    var a []int
    fmt.Println(a[10]) // 越界访问,触发panic
  3. 自定义类型错误

    通过panic抛出自定义类型(如结构体或错误接口),recover可以捕获并处理。

    go 复制代码
    panic(fmt.Errorf("custom error"))

不能recover的错误

  1. 其他goroutine未处理的panic

    每个goroutine需独立处理自己的panic。若某goroutine未在其内部defer中调用recover,该panic会导致整个程序崩溃,主goroutine无法捕获。

    go 复制代码
    go func() {
        panic("goroutine panic") // 主goroutine无法捕获此panic
    }()
  2. 程序已终止的情况

    若已调用os.Exit()或发生致命错误(如内存耗尽),recover无法恢复。

  3. defer上下文中的panic
    recover仅在defer函数中调用有效,且必须位于触发panic的同一函数调用栈中。

    go 复制代码
    func main() {
        panic("panic outside defer") // 无法被后续的defer捕获
        defer func() { recover() }() // 此处defer不会执行
    }
  4. 重复调用panic

    若在defer中再次触发panic且未被捕获,程序仍会崩溃。

    go 复制代码
    defer func() {
        recover()
        panic("re-panic") // 未被捕获,导致程序终止
    }()
    panic("initial panic")
  5. CGO或系统级错误(部分)

    某些底层系统错误(如栈溢出、内存段错误)可能绕过Go的panic/recover机制,直接终止程序。


关键规则总结

  1. 作用域限制
    recover仅在当前goroutine的defer函数中有效。

  2. 运行时错误的可恢复性

    技术上可捕获,但程序状态可能损坏,需谨慎处理。

  3. 设计建议

    • 仅对可预测的错误使用panic/recover(如输入验证失败)。
    • 避免依赖recover处理不可恢复错误(如内存越界),应确保程序健壮性。

通过合理使用deferrecover,可以优雅处理局部错误,但需理解其局限性以避免误用。

相关推荐
fliter24 分钟前
用 Rust 解析并生成 ICMP 包:checksum、nom 与 cookie-factory
后端
蝎子莱莱爱打怪26 分钟前
XZLL-IM干货系列 03|消息 ID 设计:一个 UUID 搞不定的事,我用两个 ID 解决了
后端·面试·开源
fliter26 分钟前
从 panic 到 Result:用 Rust 重新整理一个 ping 项目的错误处理
后端
森蓝情丶1 小时前
我给 AI 搭了个法庭:一个前端仔的 LangGraph 实战全记录
前端·后端
JensCS猿1 小时前
从 Spring Boot 回看 SSM 框架:手动挡与自动挡的驾驶哲学
后端
爱勇宝1 小时前
干了近 8 年,一夜之间被裁:AI 时代,程序员最该害怕的不是 AI
前端·后端·程序员
科米米1 小时前
嵌入式日志模块
后端
血小溅2 小时前
三大 AI 编码框架深度对比:GSD vs OpenSpec vs Superpowers
人工智能·后端
ThanksGive2 小时前
层级时间轮看门狗
后端
GetcharZp3 小时前
告别繁琐命令行!这款容器可视化神器,让 Docker/K8s 管理变得如此简单
后端