1 官方定義
A defer statement pushes a function call onto a list. The list of saved calls is executed after the surrounding function returns. Defer is commonly used to simplify functions that perform various clean-up actions.
defer表達(dá)式將一個函數(shù)調(diào)用保存在列表中,當(dāng)包裹defer的函數(shù)"返回"后,列表中的調(diào)用會被執(zhí)行。defer通常用于清理收尾工作。
注意:這里的返回加了引號,原因如下
2 實現(xiàn)邏輯
參考 雨痕大神的讀書筆記(https://github.com/qyuhen/book)源碼第20章
大致表達(dá)為:
step 1 : 在defer表達(dá)式的地方,會調(diào)用runtime.deferproc(size int32, fn *funcval)保存延時調(diào)用,注意這里保存了延時調(diào)用的參數(shù)
step 2 : 在return時,先將返回值保存起來
step 3 : 按FILO順序調(diào)用runtime.deferreturn,即延時調(diào)用
step 4 : RET指令
因此,return并不是一個原子操作,函數(shù)返回值可能與你的預(yù)期不一樣。
3 避坑提示
1. defer的參數(shù)在聲明時即被確定下來,先看個例子(生產(chǎn)環(huán)境這樣寫估計會被唾沫噴死)
func calc(index string, a, b int) int {
ret := a + b
fmt.Println(index, a, b, ret)
return ret
}
func main() {
a := 1
b := 2
defer calc("1", a, calc("10", a, b))
a = 0
defer calc("2", a, calc("20", a, b))
b = 1
}
輸出結(jié)果為:
10 1 2 3
20 0 2 2
2 0 2 2
1 1 3 4
原因是defer calc("1", a, calc("10", a, b)) 的第3個參數(shù)會在調(diào)用runtime.deferproc時確定,并不會在延時調(diào)用時才會被計算。
2. 有名與無名返回值
func namedReturn() (r int) {
defer func() {
r++
fmt.Println("defer in namedReturn : r = ", r)
}()
return
}
func unnamedReturn() int {
var r int
defer func() {
r++
fmt.Println("defer in unnamedReturn : r = ", r)
}()
return r
}
func main() {
fmt.Println("namedReturn : r = ", namedReturn())
fmt.Println("unnamedReturn : r = ", unnamedReturn())
}
輸出結(jié)果為:
defer in namedReturn : r = 1
namedReturn : r = 1
defer in unnamedReturn : r = 1
unnamedReturn : r = 0
原因就是return會將返回值先保存起來,對于無名返回值來說,保存在一個臨時對象中,defer是看不到這個臨時對象的;而對于有名返回值來說,就保存在已命名的變量中。
3. 延時參數(shù)與有名返回值遮蔽
func ShelteredReturn() (r int) {
defer func(r int) {
r++
fmt.Println("defer in ShelteredReturn : r = ", r)
}(r)
return 0
}
func main() {
fmt.Println("ShelteredReturn : r = ", ShelteredReturn())
}
輸出結(jié)果為:
defer in ShelteredReturn : r = 1
ShelteredReturn : r = 0
雖然r是有名返回值,但在defer func(r int)中的r是形參,與ShelteredReturn的返回值不是同一個。
參考文獻(xiàn)
[1]. http://lib.csdn.net/article/go/33950
[2]. https://blog.golang.org/defer-panic-and-recover
[3]. https://my.oschina.net/henrylee2cn/blog/505535
[4]. https://github.com/qyuhen/book