defer
关键字在go中用于return
之前执行,在golang官方文档中有说明。defer
的用法类似java中的finally
语句块,一般用于释放已分配的资源。
见如下例子:
package main
import (
"io"
"log"
)
func run(s string) (n int, err error) {
defer func() {
log.Printf("run(%q) = %d, %v", s, n, err)
}()
return 7, io.EOF
}
func main() {
run("Go")
}
输出结果:
2019/01/31 15:13:21 run("Go") = 7, EOF
那么这个结果该如何理解呢?要使用defer
时不踩坑,最重要的一点就是要明白,return xxx
这一条语句并不是一条原子指令!所以总结出返回过程如下:
- 首先执行
defer
之前对返回值赋值 - 然后调用
defer
表达式 - 调用
return
将上述规则改写成程序形式:
返回值 = xxx
执行 defer
直接return
按照这个方法改写上述程序:
func run(s string) (n int, err error) {
n = 7
err = EOF
defer func() {
log.Printf("run(%q) = %d, %v", s, n, err)
}()
return
}
所以打印出的值是run("Go") = 7, EOF
goroutine的控制结构中,有一张表记录defer
,调用runtime.deferproc
时会将需要defer
的表达式记录在表中,而在调用runtime.deferreturn
的时候,则会依次从defer表中出栈并执行。
即多个defer
调用的时候遵循后入先出原则。