面试题

这是Go Quiz系列里关于panic的第1篇,主要考察同一个goroutine在多次panic场景下recover的机制。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
// quiz0.go
package main

import "fmt"

func main() {
	defer func() { fmt.Println(recover()) }()
	defer func() { fmt.Println(recover()) }()
	defer panic(1)
	panic(2)
}
  • A: 2 <nil>
  • B: 1 <nil>
  • C: 2 1
  • D: 1 2
  • E: 直接panic

解析

defer的函数调用会被延后到函数return或者panic退出之前执行,因此本题的执行结果如下:

Step 1: 执行panic(2),触发被defer的函数的执行

Step 2: 执行代码里第9行被defer的函数调用panic(1)panic(1)会覆盖panic(2),可以当做panic(2)没有了

Step 3: 执行代码里第8行被defer的函数调用,recover()捕获panic(1),打印1

Step 4: 执行代码里第7行被defer的函数调用,recover()返回的是nil,因为panic已经被第8行的recover()捕获,所以打印nil

所以本题的答案是B

思考题

留一道思考题,想知道答案的可以给本人vx公众号发送消息panic获取答案和题目解析。

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
// quiz1.go
package main

import "fmt"

func main() {
	defer func() { fmt.Println(recover()) }()
	defer panic(1)
	panic(2)
}
  • A: 1
  • B: 2
  • C: 先打印1,然后panic
  • D: 先打印2,然后panic

加餐

开源地址

文章和示例代码开源在GitHub: Go语言初级、中级和高级教程

公众号:coding进阶。关注公众号可以获取最新Go面试题和技术栈。

个人网站:Jincheng’s Blog

知乎:无忌