簡單探討Golang中defer預計算引數
導讀 | 在golang當中defer程式碼塊會在函式呼叫連結串列中增加一個函式呼叫,下面這篇文章主要給大家介紹了關於Golang中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 是關鍵字,但是也繼承了這個特性。假設我們想要計算 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/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- 簡單聊聊Golang中defer預計算引數Golang
- 深入探討Spring Boot中的引數傳遞Spring Boot
- 簡單探討TypeScript 列舉型別TypeScript型別
- 簡單探討C#中GUI程式設計的標準事件問題C#GUI程式設計事件
- NSOperation的進階使用和簡單探討
- NSThead的進階使用和簡單探討
- 簡單探討區塊鏈中的不可能三角區塊鏈
- 深入探討:Maven中的物料清單BOMMaven
- 簡單探討JavaScript 與 TypeScript之間的聯絡JavaScriptTypeScript
- 簡單探討sum()函式返回null的問題函式Null
- 深入探討單例模式單例模式
- Golang 中 defer Close() 的潛在風險Golang
- Golang併發程式設計中select簡單瞭解Golang程式設計
- [譯] part 29: golang deferGolang
- Golang中defer的三個實戰要點Golang
- 如何計算PHP函式中傳遞的引數數量PHP函式
- golang基礎–細說deferGolang
- golang defer使用需要注意Golang
- Promise探討Promise
- Golang 中的 Defer 必掌握的 7 知識點Golang
- 簡單混合運算的計算器
- fixtrue基礎之params引數實現簡單引數化
- Dataworks批量刷數優化方案探討優化
- TRIZ理論在洗碗機設計中應用探討
- 關於python中slicing的探討Python
- XAF中XPO與EFCore的探討
- go return 的時候傳遞引數給deferGo
- 0014---簡單的計算
- ACCESS 在資料表中實現簡單計算
- golang的defer踩坑彙總Golang
- 建築師解構遊戲關卡——引數化關卡設計探討佔領據點地圖遊戲地圖
- 探尋流式計算
- golang設計模式之簡單工廠模式Golang設計模式
- 一起探討下web請求流程的程式碼結構設計(簡單以交易為栗子)Web
- 無服務計算應用場景探討及 FaaS 應用實戰
- golang flag簡單用法Golang
- 深入探討 UndefinedUndefined
- 深入探討HBASE