亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Golang Defer作用域及執(zhí)行順序使用案例

 更新時(shí)間:2023年12月18日 10:02:22   作者:程序員祝融  
這篇文章主要為大家介紹了Golang Defer作用域及執(zhí)行順序使用案例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

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)文章

  • Golang?基礎(chǔ)面試題集錦

    Golang?基礎(chǔ)面試題集錦

    這篇文章主要為大家介紹了Golang?基礎(chǔ)面試題集錦,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-09-09
  • go goroutine 怎樣進(jìn)行錯(cuò)誤處理

    go goroutine 怎樣進(jìn)行錯(cuò)誤處理

    在 Go 語言程序開發(fā)中,goroutine 的使用是比較頻繁的,因此在日常編碼的時(shí)候 goroutine 里的錯(cuò)誤處理,怎么做會比較好呢,本文就來詳細(xì)介紹一下
    2021-07-07
  • Golang創(chuàng)建第一個(gè)web項(xiàng)目(Gin+Gorm)

    Golang創(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
  • 從生成CRD到編寫自定義控制器教程示例

    從生成CRD到編寫自定義控制器教程示例

    這篇文章主要為大家介紹了從生成CRD到編寫自定義控制器的教程示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-05-05
  • Go語言讀取txt文檔的操作方法

    Go語言讀取txt文檔的操作方法

    Go語言提供了很多文件操作的支持,在不同場景下,有對應(yīng)的處理方式,本節(jié)我們來介紹一下Go語言讀取txt文檔的相關(guān)知識,感興趣的朋友跟隨小編一起看看吧
    2022-01-01
  • 詳解Golang如何監(jiān)聽某個(gè)函數(shù)的開始執(zhí)行和執(zhí)行結(jié)束

    詳解Golang如何監(jiān)聽某個(gè)函數(shù)的開始執(zhí)行和執(zhí)行結(jié)束

    這篇文章主要為大家詳細(xì)介紹了Golang如何監(jiān)聽某個(gè)函數(shù)的開始執(zhí)行和執(zhí)行結(jié)束,文中的示例代碼講解詳細(xì),有需要的小伙伴可以參考一下
    2024-02-02
  • Go?為什么不支持可重入鎖原理解析

    Go?為什么不支持可重入鎖原理解析

    這篇文章主要為大家介紹了Go?為什么不支持可重入鎖原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08
  • go語言實(shí)現(xiàn)sqrt的方法

    go語言實(shí)現(xiàn)sqrt的方法

    這篇文章主要介紹了go語言實(shí)現(xiàn)sqrt的方法,實(shí)例分析了Go語言實(shí)現(xiàn)計(jì)算平方根的技巧,需要的朋友可以參考下
    2015-03-03
  • 使用go實(shí)現(xiàn)刪除sql里面的注釋和字符串功能(demo)

    使用go實(shí)現(xiàn)刪除sql里面的注釋和字符串功能(demo)

    這篇文章主要介紹了使用go實(shí)現(xiàn)刪除sql里面的注釋和字符串功能,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-11-11
  • go-zero服務(wù)部署配置及源碼解讀

    go-zero服務(wù)部署配置及源碼解讀

    這篇文章主要為大家介紹了go-zero服務(wù)部署配置及源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08

最新評論