这两个内置函数用来处理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中被捕获,执行结果:
二、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")}()
}
执行结果:
根据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()
}