Golang 的 defer 关键字使用注意事项
Table of Contents
Golang 提供了 defer
关键字,用于函数退出前执行收尾工作,基本的使用方法就不再赘述了。总结了一些可能踩坑的地方。
defer 执行顺序
package main
import "fmt"
func TestFunc() {
defer func() {
fmt.Println("A")
}()
defer func() {
fmt.Println("B")
}()
defer func() {
fmt.Println("C")
}()
}
func main() {
TestFunc()
}
执行结果是:
C
B
A
即 defer 执行先后顺序与代码顺序是相反的,是后进先出的顺序。
defer 与 return 执行顺序
package main
import "fmt"
func deferFunc() int {
fmt.Println("defer func running")
return 0
}
func returnFunc() int {
fmt.Println("return func running")
return 0
}
func TestFunc() int {
defer deferFunc()
return returnFunc()
}
func main() {
TestFunc()
}
执行结果是:
return func running
defer func running
可以看到 return 语句先执行,defer 的函数后执行。
defer 影响返回值
既然 defer 执行的函数在 return 语句之后执行。那么 defer 是否可以改变函数返回值?来看下面的例子:
package main
import "fmt"
func TestFunc() (result int) {
defer func() {
result = 1024
}()
return 2048
}
func main() {
fmt.Println(TestFunc())
}
执行结果是 1024。return 的是 2048,即 defer 确实可以改变函数返回值。再看个类似的例子:
package main
import "fmt"
func TestFunc() int {
result := 1024
defer func() {
result = 2048
}()
return result
}
func main() {
fmt.Println(TestFunc())
}
执行结果是 1024。这时 defer func 虽然更新了 result 的值,但并没有对函数返回值造成影响。原因是该函数的 int 类型返回值未命名(即匿名返回值),所以 defer func 无法修改该返回值,只是修改了 result 变量值。
defer 函数预计算参数值
package main
import (
"fmt"
"time"
)
func TestFunc() {
startedAt := time.Now()
defer fmt.Println(time.Since(startedAt))
time.Sleep(time.Second)
}
func main() {
TestFunc()
}
这个例子是计算函数的执行时间,sleep 1s 作为模拟,然后执行结果是:199ns,显然这不是预期的结果。期望的执行时间应该是大于 1s 的。那么问题出在哪里?
由于调用 defer 时会预计算函数参数值,即预计算 time.Since(startedAt)
这个参数的值,而不是 sleep 执行完结束后计算的。可以这样改符合预期:
package main
import (
"fmt"
"time"
)
func TestFunc() {
startedAt := time.Now()
defer func() {
fmt.Println(time.Since(startedAt))
}()
time.Sleep(time.Second)
}
func main() {
TestFunc()
}
defer 后面跟匿名函数即可,符合我们的期望。执行结果是:1.003861399s。
defer 与 panic
触发 defer 执行的时机有三个:
- 包裹 defer 的函数遇到 return
- 包裹 defer 的函数执行到最后
- 发生 panic
来看下 defer 与 panic 相遇时会发生什么。
发生 panic 时在 defer 中不捕获异常
package main
import "fmt"
func TestFunc() {
defer func() {
fmt.Println("A")
}()
defer func() {
fmt.Println("B")
}()
panic("panic occurred")
}
func main() {
TestFunc()
}
执行结果:
B
A
panic: panic occurred
... // panic 堆栈信息
发生 panic 时在 defer 中捕获异常
package main
import "fmt"
func TestFunc() {
defer func() {
fmt.Println("A")
}()
defer func() {
fmt.Println("B")
if err := recover(); err != nil {
fmt.Println("catch panic")
}
}()
panic("panic occurred")
}
func main() {
TestFunc()
}
执行结果是:
B
catch panic
A
这个例子执行结果没什么可说的,发生 panic,可以在 defer 函数中捕获处理。
defer 中发生 panic
package main
import "fmt"
func TestFunc() {
defer func() {
if err := recover(); err != nil {
fmt.Printf("catch %s panic\n", err)
}
}()
defer func() {
panic("inner")
}()
panic("outer")
}
func main() {
TestFunc()
}
执行结果是:catch inner panic。即外层发生 panic(“outer”),触发 defer 执行,执行 defer 中又发生 panic(“inner”),最后一个 defer 中捕获 panic,捕获到的是最后一个 panic。
另外,尽量不要使用 panic,仅在发生不可恢复的错误或程序启动时发生意外才使用 panic。