Go: defer與return小記

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

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 國家電網(wǎng)公司企業(yè)標(biāo)準(zhǔn)(Q/GDW)- 面向?qū)ο蟮挠秒娦畔?shù)據(jù)交換協(xié)議 - 報批稿:20170802 前言: 排版 ...
    庭說閱讀 12,306評論 6 13
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 178,777評論 25 709
  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 2,030評論 0 9
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 136,506評論 19 139
  • 我們村有條河自東向西流過。說是河,但其實很多時候它只能算作寬廣的小溪,只有在地勢突然降低的時候,河水才陡的變深。 ...
    官官抓鬮閱讀 336評論 0 1

友情鏈接更多精彩內(nèi)容