簡單探討Golang中defer預計算引數

大雄45發表於2023-01-02
導讀 在golang當中defer程式碼塊會在函式呼叫連結串列中增加一個函式呼叫,下面這篇文章主要給大家介紹了關於Golang中defer預計算引數的相關資料,文中透過例項程式碼介紹的非常詳細,需要的朋友可以參考下
什麼是defer

defer用來宣告一個延遲函式,把這個函式放入到一個棧上, 當外部的包含方法return之前,返回引數到呼叫方法之前呼叫,也可以說是執行到最外層方法體的"}"時呼叫。我們經常用他來做一些資源的釋放,比如關閉io操作

func doSomething(fileName string) {
    file,err := os.Open(fileName)
    if err != nil {
    panic(err)
    }
    defer file.Close()
}

defer 可以保證方法可以在外圍函式返回之前呼叫。有點像其他言的 try finally

try{
}finally{
}
Go語言defer預計算引數

Go 語言中所有的函式呼叫都是傳值的,雖然 defer 是關鍵字,但是也繼承了這個特性。假設我們想要計算 main 函式執行的時間,可能會寫出以下的程式碼:

package main
import (
    "fmt"
    "time"
)
 
func main() {
    startedAt := time.Now()
    defer fmt.Println(time.Since(startedAt))
    time.Sleep(time.Second) //休眠一秒
}

結果是:

D:\workspace\go\src\test>go run main.go
0s

執行結果並不符合我們的預期,這個現象背後的原因是什麼呢?經過分析,我們會發現呼叫 defer 關鍵字會立刻複製函式中引用的外部引數,所以 time.Since(startedAt) 的結果不是在 main 函式退出之前計算的,而是在 defer 關鍵字呼叫時計算的【defer入棧的時候】,最終導致上述程式碼輸出 0s

我們再來看個簡單例子來說明上述解釋:

package main
import (
    "fmt"
)
 
func main() {
    i := 1
    defer fmt.Println(test(i))
    i = 100
}
 
func test(i int) int {
    i = i + 1
    return i
}
D:\workspace\go\src\test>go run main.go
2

當程式碼執行到defer fmt.Println(test(i))的時候,會把defer右邊最外層函式的引數計算完畢,並傳遞進函式里,但不會執行函式體的程式碼直到包裹defer的函式返回。我們先看會把defer右邊最外層函式的引數計算完畢,並傳遞進函式里這句話,對應例子就是先把test(i)算出來,此時i=1,計算test(1)得2,然後fmt.Println(2)入棧,等到最後程式執行完了再執行defer結果就是2(但不會執行函式體的程式碼直到包裹defer的函式返回)。

我們再來看一個例子與匿名函式結合:

package main
import (
    "fmt"
)
 
func main() {
    i := 1
    defer func() {
        fmt.Println(test(i))
    }()
    i = 100
}
 
func test(i int) int {
    i = i + 1
    return i
}

結果:

D:\workspace\go\src\test>go run main.go
101

使用匿名函式,結果是101,相當於i給到test方法的是100,那為什麼呢?還是那句話:但不會執行函式體的程式碼直到包裹defer的函式返回

也就是說他會把整個{ fmt.Println(test(i)) }()函式體入棧,等到最後程式執行完了再執行defer,此時的i是100,執行test後就是101了。

所以你要解決第一個列印為0s的問題,你就可以使用匿名函式來解決,如下:

package main
import (
    "fmt"
    "time"
)
 
func main() {
    startedAt := time.Now()
    defer func() {
        fmt.Println(time.Since(startedAt))
    }()
    time.Sleep(time.Second) //休眠一秒
}

結果:

D:\workspace\go\src\test>go run main.go
1.0152825s

到此這篇關於Golang中defer預計算引數的文章就介紹到這了。

原文來自:

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/69955379/viewspace-2886369/,如需轉載,請註明出處,否則將追究法律責任。

相關文章