当前位置: 代码迷 >> 综合 >> golang-panic和recover
  详细解决方案

golang-panic和recover

热度:59   发布时间:2023-10-13 09:21:42.0

这两个内置函数用来处理go的运行时错误,panic用来主动抛出错误,recover用来捕获错误。

一、发生panic后,函数会从调用panic的位置返回,逐层向上执行函数的defer语句,直到被recover捕获或运行到最外层函数退出。

package mainimport "fmt"
func f3(){
    panic("aaa")
}
func f4(){
    f3()
}
func main(){
    defer func(){
    fmt.Println("a")fmt.Println(recover())}()f4()}

f3()中的panic向上经过f4(),在main函数的defer中被捕获,执行结果:
golang-panic和recover
二、recover()只有在defer后面的函数体内被直接调用才能捕获panic异常

//无效
//(1)
recover()
//(2)
defer recover()
//(3)
defer fmt.Println(recover())
//(4)嵌套两层函数,无效
defer func(){
    func(){
    recover()}
}
//有效
//(1)
defer func(){
    fmt.Println("a")fmt.Println(recover())}()
//(2)
func f1(){
    fmt.Println(recover())
}
defer func(){
    fmt.Println("a")f1()	}()

三、可以有多个panic被抛出(只能出现在延迟调用里面,且只有最后一个panic能被捕获)

package mainimport "fmt"func main(){
    defer func(){
    fmt.Println("a")fmt.Println(recover())}()defer func(){
    fmt.Println("b")fmt.Println(recover())}()defer func(){
    panic("aaa")}()defer func(){
    panic("bbb")}()defer func(){
    panic("ccc")}()
}

执行结果:
golang-panic和recover根据defer先进后出的规则只有

	defer func(){
    panic("aaa")}()

的异常被捕获,且只有

	defer func(){
    fmt.Println("b")fmt.Println(recover())}()

捕获到异常。

四、包含init()函数引发的panic只能在init函数中捕获,无法在main函数中被捕获,因为init函数优先于main函数

package mainimport "fmt"
func f5(){
    panic("aaa")
}
func init(){
    defer func(){
    fmt.Println(recover())}()f5()
}
func main(){
    }

五、函数不能捕获内部协程所抛出的panic

package mainimport "fmt"func f5(){
    panic("aaa")
}
func main(){
    defer func(){
    fmt.Println(recover())}()go f5()
}