这两个内置函数用来处理go的运行时错误,panic用来主动抛出错误,recover用来捕获错误。
一、发生panic后,函数会从调用panic的位置返回,逐层向上执行函数的defer语句,直到被recover捕获或运行到最外层函数退出。
package main
import "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 main
import "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 main
import "fmt"
func f5(){
panic("aaa")
}
func init(){
defer func(){
fmt.Println(recover())
}()
f5()
}
func main(){
}
五、函数不能捕获内部协程所抛出的panic
package main
import "fmt"
func f5(){
panic("aaa")
}
func main(){
defer func(){
fmt.Println(recover())
}()
go f5()
}