Golang Defer作用域及執(zhí)行順序使用案例
1 defer 作用
在 Golang 中,defer 比面向?qū)ο笳Z言中的析構(gòu)函數(shù)要強(qiáng)大很多,defer 還有錯(cuò)誤捕獲、修改函數(shù)返回值、資源釋放等,defer 會在當(dāng)前所在函數(shù)返回前執(zhí)行傳入的函數(shù)。例如:
func CreateUsers(db *gorm.DB) error { tx := db.Begin() defer tx.Rollback() if err := tx.Create(&Users{Name: "祝融"}).Error; err != nil { return err } return tx.Commit().Error }
注:調(diào)用 tx.Commit()
之后執(zhí)行 tx.Rollback()
并不會影響已經(jīng)提交事務(wù)。
2 defer 作用域
defer 作用域在當(dāng)前函數(shù)和方法返回之前被調(diào)用。例如:
package main import "fmt" func main() { { defer fmt.Println("defer done") fmt.Println("code block done") } fmt.Println("main done...") } $ go run main code block done main done defer done
我們會發(fā)現(xiàn),傳入的函數(shù)不是在退出代碼塊的作用域時(shí)執(zhí)行的,defer 只會在當(dāng)前函數(shù)和方法返回之前被調(diào)用。
3 defer 執(zhí)行順序
在 Golang 中,defer 的執(zhí)行順序是采用棧(stack)的方式。當(dāng)你使用多個(gè) defer 語句時(shí),它們會按照后進(jìn)先出(LIFO)的順序執(zhí)行。在一個(gè)函數(shù)生命周期內(nèi),優(yōu)先調(diào)用后面的 defer 。例如:
package main import "fmt" func main() { defer funcA() defer funcB() defer funcC() } func funcA() { fmt.Println("A") } func funcB() { fmt.Println("B") } func funcC() { fmt.Println("C") } $ go run main.go C B A
圖解:
3.1 defer、return 誰先執(zhí)行?
在 Golang 中,return 比 defer 先執(zhí)行,例如:
package main import "fmt" func deferFunc() int { fmt.Println("defer func done") return 0 } func returnFunc() int { fmt.Println("return func done") return 0 } func returnAndDefer() int { defer deferFunc() return returnFunc() } func main() { returnAndDefer() } $ go run main.go return func done defer func done
3.2 defer 影響主函數(shù)的具名返回值
上面講了 return 比 defer 先執(zhí)行。
當(dāng)主函數(shù)有返回值,且返回值有沒有名字沒有關(guān)系,defer 所作用的函數(shù),即 defer 可能會影響主函數(shù)返回值??匆粋€(gè)例子:
func main() { fmt.Println(deferFuncReturn()) } func deferFuncReturn() (j int) { // t初始化0且作用域?yàn)樵摵瘮?shù)全域 i := 1 defer func() { j++ }() return i } $ go run main.go 2
在舉一個(gè)例子對比一下:
func main() { fmt.Println(deferFuncReturn()) } func deferFuncReturn() int { i := 1 defer func() { i++ }() return i } $ go run main.go 1
結(jié)論:當(dāng)主函數(shù)有返回值 ,會在函數(shù)初始化時(shí)賦值為0,且其作用域?yàn)樵摵瘮?shù)全域,defer 會影響到該返回值。
3.3 defer 偶遇 panic
在 Golang 中,執(zhí)行過程中遇到 panic 錯(cuò)誤時(shí),遍歷所有defer,強(qiáng)行 defer 出棧,并執(zhí)行 defer。在執(zhí)行過程中,
- 遇到 recover 捕獲異常停止 panic,返回 recover 繼續(xù)執(zhí)行
- 未設(shè)置 recover 捕獲異常,遍歷完 defer 拋出 panic 信息
3.3.1 defer 未捕獲 panic
package main import ( "fmt" ) func main() { defer_call() fmt.Println("main done...") } func defer_call() { defer func() { fmt.Println("defer func 1") }() defer func() { fmt.Println("defer func 2") }() // 觸發(fā)defer出 panic("error") defer func() { fmt.Println("defer func 3: no exec") }() } $ go run main.go defer func 2 defer func 1 panic:error ... 堆棧error...
3.3.2 defer 捕獲 panic
package main import ( "fmt" ) func main() { defer_call() fmt.Println("main done...") } func defer_call() { defer func() { fmt.Println("defer func 1, 捕獲異常") if err := recover(); err != nil { fmt.Println(err) } }() defer func() { fmt.Println("defer func 2, 沒有捕獲異常") }() // 觸發(fā)defer出 panic("error") defer func() { fmt.Println("defer func 3: no exec") }() } $ go run main.go defer func 2, 沒有捕獲異常 defer func 1, 捕獲異常 error main done...
總結(jié):從上面可以看出,程序執(zhí)行中發(fā)生了 panic 異常,panic 前的 defer 一定能被執(zhí)行到,所以我們一般用于關(guān)閉資源等,這樣一定能保證資源能被關(guān)閉,避免一下問題。
3.3.3 defer 中含有 panic
Golang 中,panic 僅會被最后一個(gè) revover 捕獲。
package main import ( "fmt" ) func main() { defer func() { if err := recover(); err != nil{ fmt.Println("err:", err) }else { fmt.Println("fatal") } }() defer func() { panic("defer panic2") }() panic("panic1") } $ go run main.go err: defer panic2
在上面例子中,panic("panic1")
先 觸發(fā) defer 強(qiáng)制出棧,第一個(gè)執(zhí)行觸發(fā) panic("defer panic2)"
異常,此時(shí)會覆蓋前一個(gè)異常 panic
,最后繼續(xù)執(zhí)行 defer, 最終被 recover()
捕獲住。
3.4 defer 函數(shù)嵌套子函數(shù)
分析下方代碼,有 4 個(gè)函數(shù),其中 x 為 1、2、3、4
package main import "fmt" func f(x int, y int) int { fmt.Println("x:", x) return x } func main() { defer f(1, f(3, 0)) defer f(2, f(4, 0)) }
先分析下執(zhí)行順序,有 2 個(gè) defer,則會產(chǎn)生 2 次入棧操作,分別是 f1 、f2。
- f1 入棧時(shí),因?yàn)樾螀?y 是一個(gè)函數(shù),則需要執(zhí)行該函數(shù),故執(zhí)行一次輸出 (x:3)
- f2 入棧時(shí),因?yàn)樾螀?y 是一個(gè)函數(shù),則需要執(zhí)行該函數(shù),故執(zhí)行一次輸出(x:4)
main 函數(shù)執(zhí)行完后,執(zhí)行 defer 函數(shù),所有 defer 出棧,所有執(zhí)行順序?yàn)?f2、f1。所以程序最終輸出的結(jié)果是:
$ go run main.go 3 4 2 1
4 思考
一下所有函數(shù)傳參均為 1。
4.1 defer_fun1
考點(diǎn):res 的作用域
func defer_fun1(x int) (res int) { res = x defer func() { res += 3 }() return res }
題解:
- 函數(shù)有返回值,res 初始化為 0
- res = x 則 res = 1,defer 入棧
- return res 函數(shù)結(jié)束后,defer 出棧執(zhí)行 res +3 = 4
最終函數(shù)返回結(jié)果為 4
4.2 defer_fun2
func defer_fun2(x int) int { res := x defer func() { res += 3 }() return res }
題解:
- res = x 則 res = 1,defer 入棧
- return res 程序結(jié)束后,此時(shí)函數(shù)返回值為1 ,但是返回值不是 res。
- defer 出棧執(zhí)行 res +3 = 4
最終函數(shù)返回結(jié)果為 1
4.3 defer_fun3
func defer_fun3(x int) (res int) { defer func() { res += x }() return 2 }
題解:
- 函數(shù)有返回值,res 初始化為 0
- x = 1, defer 入棧
- 函數(shù)結(jié)束res=2,執(zhí)行 defer 出棧執(zhí)行 res +1 = 3
最終函數(shù)返回結(jié)果為 3
4.4 defer_fun4
func defer_fun4() (res int) { t := 1 defer func(x int) { fmt.Println("x:", x) fmt.Println("res:", res) }(t) t = 2 return 4 }
題解:
- 函數(shù)有返回值,res 初始化為 0
- t 初始化為1
- defer 入棧,x 作為形參
- 執(zhí)行t =2,此時(shí)主函數(shù)返回 4,則 res = 4
- 最后 defer 出棧,控制臺輸出 x: 1 res: 4
最終函數(shù)返回結(jié)果為 4。
以上就是Golang Defer作用域及執(zhí)行順序使用案例的詳細(xì)內(nèi)容,更多關(guān)于Golang Defer作用域的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
go goroutine 怎樣進(jìn)行錯(cuò)誤處理
在 Go 語言程序開發(fā)中,goroutine 的使用是比較頻繁的,因此在日常編碼的時(shí)候 goroutine 里的錯(cuò)誤處理,怎么做會比較好呢,本文就來詳細(xì)介紹一下2021-07-07Golang創(chuàng)建第一個(gè)web項(xiàng)目(Gin+Gorm)
本文主要介紹了Golang創(chuàng)建第一個(gè)web項(xiàng)目(Gin+Gorm),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-06-06詳解Golang如何監(jiān)聽某個(gè)函數(shù)的開始執(zhí)行和執(zhí)行結(jié)束
這篇文章主要為大家詳細(xì)介紹了Golang如何監(jiān)聽某個(gè)函數(shù)的開始執(zhí)行和執(zhí)行結(jié)束,文中的示例代碼講解詳細(xì),有需要的小伙伴可以參考一下2024-02-02使用go實(shí)現(xiàn)刪除sql里面的注釋和字符串功能(demo)
這篇文章主要介紹了使用go實(shí)現(xiàn)刪除sql里面的注釋和字符串功能,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-11-11