7.4. 第3节:recover

未匹配的标注
本文档最新版为 2023,旧版本可能放弃维护,推荐阅读最新版!

recover

运行时panic异常一旦被引发就会导致程序崩溃。这当然不是我们愿意看到的,因为谁也不能保证程序不会发生任何运行时错误。

Go语言为我们提供了专用于“拦截”运行时panic的内建函数——recover。它可以是当前的程序从运行时panic的状态中恢复并重新获得流程控制权。

语法如下:
func recover() interface{}
注意:recover只有在defer调用的函数中有效。

示例如下:

func TestA() {
    fmt.Println("func TestA()")
}
func TestB() {
    // 设置recover
    defer func() {
        recover()
    }() // 自调用改匿名函数
    var a [10] int
    a[x] = 111 // 当x为20 的时候,导致数组越界,产生pani,导致程序崩溃
}
func TestC() {
    fmt.Println("func TestC()")
}

func main() {
    TestA()
    TestB(11)  // TestB() 发生异常 中断程序
    TestC()
}

以上程序的运行结果如下:

第3节:recover

通过以上程序,我们发现虽然TestB( )函数会导致整个应用程序崩溃,但是由于在改函数中调用了recover( )函数,所以整个函数并没有崩溃。虽然程序没有崩溃,但是我们也没有看到任何的提示信息,那么怎样才能够看到相应的提示信息呢?

可以直接打印recover( )函数的返回结果,如下所示:

func TestB() {
    // 设置recover
    defer func() {
        // recover()
        // 打印 pani的错误信息
        fmt.Println(recover())
    }() // 自调用改匿名函数
    var a [10] int
    a[x] = 111 // 当x为20 的时候,导致数组越界,产生panic导致程序崩溃
}

输出结果如下:

第3节:recover

从输出结果发现,确实打印出了相应的错误信息。
但是,如果程序没有出错,也就是数组下标没有越界,会出现什么情况呢?

func TestA() {
    fmt.Println("func TestA()")
}
func TestB() {
    // 设置recover
    defer func() {
        recover()
    }() // 自调用改匿名函数
    var a [10] int
    a[x] = 111 // 当x为20 的时候,导致数组越界,产生panic导致程序崩溃
}
func TestC() {
    fmt.Println("func TestC()")
}

func main() {
    TestA()
    TestB(1)  // TestB() 不让他越界
    TestC()
}

输出如下:

第3节:recover

这时输出的是空,但是我们希望程序没有错误的时候,不输出任何内容。

所以,程序修改如下:
其实就是加了一层判断。

func TestA() {
    fmt.Println("func TestA()")
}
func TestB() {
    // 设置recover
    defer func() {
        if err := recover(); err != nil { // 产生了panic 异常
            fmt.Println(err)
        }
    }() // 自调用改匿名函数
    var a [10] int
    a[x] = 111 // 当x为20 的时候,导致数组越界,产生panic,导致程序崩溃
}
func TestC() {
    fmt.Println("func TestC()")
}

func main() {
    TestA()
    TestB(1)  // TestB() 不让他越界
    TestC()
}

本文章首发在 LearnKu.com 网站上。

上一篇 下一篇
讨论数量: 0
发起讨论 只看当前版本


暂无话题~