首頁 > 軟體

簡單聊聊Golang中defer預計算引數

2022-03-25 19:00:19

什麼是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:workspacegosrctest>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:workspacegosrctest>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:workspacegosrctest>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:workspacegosrctest>go run main.go
1.0152825s

總結

到此這篇關於Golang中defer預計算引數的文章就介紹到這了,更多相關Go defer預計算引數內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com