Golang 之 defer

來源:互聯網
上載者:User
這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。

關於延時調用函數(Deferred Function Calls)

延時調用函數的文法如下:

defer func_name(param-list)

當一個函數調用前有關鍵字 defer 時, 那麼這個函數的執行會延遲到包含這個 defer 語句的函數即將返回前才執行. 例如:

func main() {    defer fmt.Println("Fourth")    fmt.Println("First")    fmt.Println("Third")}

最後列印順序如下:

FirstSecondThird

需要注意的是, defer 調用的函數參數的值 defer 被定義時就確定了.
例如:

i := 1defer fmt.Println("Deferred print:", i)i++fmt.Println("Normal print:", i)

列印的內容如下:

Normal print: 2Deferred print: 1

因此我們知道, 在 "defer fmt.Println("Deferred print:", i)" 調用時, i 的值已經確定了, 因此相當於 defer fmt.Println("Deferred print:", 1) 了.
需要強調的時, defer 調用的函數參數的值在 defer 定義時就確定了, 而 defer 函數內部所使用的變數的值需要在這個函數運行時才確定. 例如:

func f1() (r int) {    r = 1    defer func() {        r++        fmt.Println(r)    }()    r = 2    return}func main() {    f1()}

上面的例子中, 最終列印的內容是 "3", 這是因為在 "r = 2" 賦值之後, 執行了 defer 函數, 因此在這個函數內, r 的值是2了, 自增後變為3.

defer 順序

如果有多個defer 調用, 則調用的順序是先進後出的順序, 類似於入棧出棧一樣:

func main() {    defer fmt.Println(1)    defer fmt.Println(2)    defer fmt.Println(3)    defer fmt.Println(4)}

最先執行的是 "fmt.Println(4)", 接著是 "fmt.Println(3)" 依次類推, 最後的輸出如下:

4321

defer 注意要點

defer 函數調用的執行時機是外層函數設定傳回值之後, 並且在即將返回之前.
例如:

func f1() (r int) {    defer func() {        r++    }()    return 0}func main() {    fmt.Println(f1())}

上面 fmt.Println(f1()) 列印的是什麼呢? 很多朋友可能會認為列印的是0, 但是正確答案是 1. 這是為什麼呢?
要弄明白這個問題, 我們需要牢記兩點

  • defer 函數調用的執行時機是外層函數設定傳回值之後, 並且在即將返回之前
  • return XXX 操作並不是原子的.

我們將上面的例子改寫一下大家就很明白了:

func f1() (r int) {    defer func() {        r++    }()    r = 0    return}

當進行賦值操作 "r = 0" 後, 才調用 defer 函數, 最後才是返回語句.
因此上面的代碼等效於:

func f1() (r int) {    r = 0    func() {        r++    }()    return}

接下來我們再來看一個更有意思的例子:

func double(x int) int {    return x + x}func triple(x int) (r int) {    defer func() {        r += x    }()    return double(x)}func main() {    fmt.Println(triple(3))}

如果我們已經理解了上面所說的內容的話, 那麼 triple 函數就很好理解了, 它實際上是:

func triple(x int) (r int) {    r = double(x)    func() {        r += x    }()    return}

defer 運算式的使用情境

defer 通常用於 open/close, connect/disconnect, lock/unlock 等這些成對的操作, 來保證在任何情況下資源都被正確釋放. 在這個角度來說, defer 操作和 Java 中的 try ... finally 語句塊有異曲同工之處.
例如:

var mutex sync.Mutexvar count = 0func increment() {    mutex.Lock()    defer mutex.Unlock()    count++}

在increment 函數中, 我們為了避免競態條件的出現, 而使用了 Mutex 進行加鎖. 而在進行並發編程時, 加鎖了卻忘記(或某種情況下 unlock 沒有被執行), 往往會造成災難性的後果. 為了在任意情況下, 都要保證在加鎖操作後, 都進行對應的解鎖操作, 我們可以使用 defer 調用解鎖操作.

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.