一、今日题目
-
下面列举的是recover()的几种调用方式,哪些是正确的?
A.func A() { recover() panic(1) }
B.
func main() { defer recover() panic(1) }
C.
func main() { defer func() { recover() }() panic(1) }
D.
func main() { defer func() { defer func() { recover() }() }() panic(1) }
-
下面代码会输出什么,请说明?
func main() { defer func() { fmt.Print(recover()) }() defer func() { defer fmt.Print(recover()) panic(1) }() defer recover() panic(2) }
二、答案:
- C
- 21
三、解析:
- recover() 必须在defer函数中直接调用才有效。A、B和D这几种情况的调用都是无效的:直接调用recover()、在defer()中直接调用recover()和defer()调用时多层嵌套。
- recover() 必须在defer()函数中调用才有效,所以第9行代码捕获是无效的。在调用defer()时,便会计算函数的参数并压入栈中,defer()的执行顺序为:先进后出。最后panic(2)会向上传递到第二个defer()中。所以在执行第6行代码时,会捕获panic(2);此后的panic(1),继续向上传递会被上一层的recover()捕获。
四、相关知识总结:
1. panic()函数
panic是用来表示非常严重的且不可恢复的错误的。在Go语言中这是个内置函数,接受一个interface{}类型的值(也就是任何值)作为参数。panic的作用就是异常,不过Go中没有try…catch,所以panic一般会导致程序挂掉(除非有recover)。
注意:即使函数执行的时候panic了,函数不往下走了,运行时并不是立刻向上传递panic,而是到defer那,等defer的东西都跑完了,panic再往上传递。
2.recover()函数
Go语言提供recover内置函数,用来捕获抛出的异常panic。
需要注意的是:必须要先声明defer,否则不能捕获到panic异常。正如上面说到的 ,recover()必须再defer函数中直接调用才有效。不使用defer,直接defer recover(),或者嵌套使用defer 函数调用都是无效的。
recover()捕获异常的机制:一旦遇到panic,panic函数不会立刻就返回,而是逻辑走到defer那,然后我们就在defer这调用recover函数就会捕获当前的panic(如果有的话),然后将defer执行完。但是recover之后逻辑不会再恢复到panic那个点去。函数会在defer之后返回。这时捕获的panic就不会向上传递了,程序就可以正常运行了。
实例:
func main() {
defer func() {
fmt.Println("c")
fmt.Println(recover())
fmt.Println("d")
}()
f()
}
func f() {
fmt.Println("a")
panic(55)
fmt.Println("b")
fmt.Println("f")
}
输出结果:
a
c
55
d