欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Golang關(guān)鍵字defer的用法詳解

 更新時(shí)間:2023年05月05日 10:10:21   作者:本生來(lái)滑稽  
defer是Go里面的一個(gè)關(guān)鍵字,用在方法或函數(shù)前面,作為方法或函數(shù)的延遲調(diào)用。這篇文章主要為大家介紹了defer的簡(jiǎn)單使用,需要的可以參考一下

1. defer的簡(jiǎn)單介紹與使用場(chǎng)景

defer是Go里面的一個(gè)關(guān)鍵字,用在方法或函數(shù)前面,作為方法或函數(shù)的延遲調(diào)用。它主要用于以下兩個(gè)場(chǎng)景

  • 優(yōu)雅釋放資源,比如一些網(wǎng)絡(luò)連接、數(shù)據(jù)庫(kù)連接以及文件資源的釋放。
  • 與recover配合處理panic異常

場(chǎng)景一:復(fù)制文件

func CopyFile(dstFile, srcFile string) (wr int64, err error) { 
    src, err := os.Open(srcFile)     
    if err != nil {         
        return     
    }      
    dst, err := os.Create(dstFile)     
    if err != nil {         
        return    
    }      
    wr, err = io.Copy(dst, src)     
    dst.Close()     
    src.Close()     
    return 
}

這樣的代碼是有問(wèn)題的,當(dāng)?shù)?行執(zhí)行失敗的時(shí)候程序返回但沒(méi)有關(guān)閉前面成功打開(kāi)的src,資源沒(méi)有正確關(guān)閉,正確代碼如下:

(在成功打開(kāi)資源,沒(méi)返回err的情況下,都可以使用defer進(jìn)行優(yōu)雅的關(guān)閉資源)

func CopyFile(dstFile, srcFile string) (wr int64, err error) {
    src, err := os.Open(srcFile)
    if err != nil {
        return
    }
    defer src.Close()
    dst, err := os.Create(dstFile)
    if err != nil {
        return
    }
    defer dst.Close()
    wr, err = io.Copy(dst, src)  
    return err
}

場(chǎng)景二:處理異常panic

package main
import "fmt"
func main() {
   defer func() {
      if r := recover(); r != nil {
         fmt.Println(r)
      }
   }()
   a := 1
   b := 0
   fmt.Println("result:", a/b)   
}

運(yùn)行結(jié)果:

runtime error: integer divide by zero

程序沒(méi)有輸出result, 會(huì)拋出panic, 因?yàn)椴荒軐?duì)除數(shù)為0的數(shù)做除法,我們使用defer在程序發(fā)生panic的時(shí)候捕獲異常。go中是用panic拋異常,recover捕獲異常,異常會(huì)在下一篇go文章進(jìn)行分析。

通過(guò)這兩個(gè)使用場(chǎng)景我們也可以看到defer后面跟著的函數(shù)被調(diào)用的時(shí)間:

  • 函數(shù)return的時(shí)候
  • 當(dāng)前協(xié)程發(fā)生panic的時(shí)候

說(shuō)到延遲函數(shù)被調(diào)用的時(shí)機(jī),這時(shí)順便說(shuō)一下多個(gè)延遲函數(shù)被調(diào)用時(shí)候的執(zhí)行順序問(wèn)題。官方對(duì)defer的解釋中寫(xiě)到每次defer語(yǔ)句執(zhí)行的時(shí)候,會(huì)把函數(shù)“壓棧”,同時(shí)函數(shù)參數(shù)會(huì)被拷貝下來(lái)

這兩點(diǎn)很重要:

一是說(shuō)明當(dāng)一個(gè)函數(shù)中有多個(gè)defer的時(shí)候,執(zhí)行順序是LIFO先進(jìn)后出

二是說(shuō)明延遲函數(shù)的參數(shù)在defer語(yǔ)句出現(xiàn)時(shí)就已經(jīng)確定下來(lái)了

這段代碼是用來(lái)補(bǔ)充這二點(diǎn)的:看一下能否看懂執(zhí)行結(jié)果,能的話就直接跳到第二部分

func deferRun1() {
   var num = 1
   numptr := &num
   defer fmt.Println("defer run 1: \n", numptr, num, *numptr) // 0xc000022088 1 1
   defer func() {
      fmt.Println("defer run 2 : \n", numptr, num, *numptr) // 0xc000116028 2 2
   }()
   defer func(num int, numptr *int) {
      fmt.Println("defer run 3: \n", numptr, num, *numptr) //0xc000116028 1 2
   }(num, numptr)
   num = 2
   fmt.Println(numptr, num, *numptr) // 0xc000116028 2 2
   return
}

對(duì)于第一個(gè)defer,傳入的num和*numptr都是一個(gè)具體的值,所以程序return完之后結(jié)果會(huì)是1,1

對(duì)于第二個(gè)defer, 沒(méi)有傳入?yún)?shù),結(jié)果會(huì)和最后的num和numptr地址對(duì)應(yīng)的內(nèi)容相同

對(duì)于第三個(gè)defer, 傳入的參數(shù)num是固定的值,而numptr是固定的地址,后面地址對(duì)應(yīng)的內(nèi)容被修改了,所以結(jié)果是1,2

同時(shí),執(zhí)行順序會(huì)是第三個(gè)defer先執(zhí)行,然后是第二個(gè)...第一個(gè)

2. defer在return執(zhí)行的時(shí)機(jī)

第一小節(jié)上面說(shuō)了有一種情況延遲函數(shù)的執(zhí)行是在return的時(shí)候,再具體一點(diǎn)就是在return的時(shí)候,defer操作帶來(lái)了什么結(jié)果呢?

一句話解決這個(gè)問(wèn)題就是:函數(shù)return不是一個(gè)原子操作,需要經(jīng)過(guò)以下三步:

  • 設(shè)置返回值
  • 執(zhí)行defer
  • 將結(jié)果返回

一定要牢記在心中,分析的時(shí)候也要嚴(yán)格按照這三步來(lái),否則極容易掉坑!

下面把需要注意的情況列舉出來(lái):

情況一:延遲函數(shù)參數(shù)早已固定下來(lái)(第一小節(jié)提到的重要的一點(diǎn))

func main() {
   deferRun()
   deferRun2()
}
func deferRun() {
   var num = 1
   defer fmt.Printf("num is %d\n", num)
   num = 2
   return
}
func deferRun2() {
   var arr = [4]int{1, 2, 3, 4}
   defer printArr(&arr)
   arr[0] = 100
   return
}
func printArr(arr *[4]int) {
   for i := range arr {
      fmt.Println(arr[i])
   }
}

運(yùn)行結(jié)果為 num is 1 以及 100,2,3,4

道理很簡(jiǎn)單:延遲函數(shù)的參數(shù)在defer出現(xiàn)的時(shí)候就固定了,對(duì)于deferRun,傳的參數(shù)是num的值,而對(duì)于deferRun,傳的參數(shù)是arr的地址,地址不變,但是地址對(duì)應(yīng)的內(nèi)容被修改,所以輸出被改變。

情況二: 嚴(yán)格把握三步走,繞開(kāi) “匿名” 小坑,分析看注釋

func main() {
   res1 := deferRun()
   fmt.Println(res1)
   res2 := deferRun2()
   fmt.Println(res2)
   res3 := deferRun3()
   fmt.Println(res3)
   res4 := deferRun4()
   fmt.Println(res4)
}
// return 2 
func deferRun() (res int) {
   num := 1
   defer func() {
      res++
   }()
   return num   //1.返回值參數(shù)res賦值為num即1  //2.執(zhí)行defer語(yǔ)句,res++  //3. 結(jié)果返回2
}
// return 1
func deferRun2() int {   // 注意這里返回值是匿名的,我們可以在心里為這個(gè)匿名的返回值取名為res
   var num int
   defer func() {
      num++
   }()
   return 1  //1.匿名返回值參數(shù)(res)賦值為1  //2.執(zhí)行defer語(yǔ)句,num++,加了個(gè)寂寞 //3.結(jié)果返回1
}
//return 1
func deferRun3() int {
   num := 1
   defer func() {
      num++
   }()
   return num  //1.匿名(res)賦值為num即1  //2.執(zhí)行defer語(yǔ)句,num++,加了個(gè)寂寞,num為2,res依舊為1 
              //3.結(jié)果返回1
}
//return 2
func deferRun4() (res int) {
   num := 1
   defer func() {
      res++
   }()
   return num //1.返回值參數(shù)res賦值為1  //2.執(zhí)行defer語(yǔ)句,res++ //3.結(jié)果返回2
}

執(zhí)行結(jié)果為 2 1 1 2

3. 小結(jié)

defer很優(yōu)雅,可以很簡(jiǎn)潔的釋放資源同時(shí)也可以與recover配合進(jìn)行panic異常的處理

defer定義的延遲函數(shù)的參數(shù)在defer語(yǔ)句出現(xiàn)的時(shí)候就確定下來(lái)了,注意有時(shí)候確定的是地址

return不是原子操作,包括了三個(gè)重要的步驟,順序是:(注意匿名返回值參數(shù))

設(shè)置返回值參數(shù)——>執(zhí)行defer語(yǔ)句——>返回結(jié)果

到此這篇關(guān)于Golang關(guān)鍵字defer的用法詳解的文章就介紹到這了,更多相關(guān)Golang關(guān)鍵字defer內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 服務(wù)器端Go程序?qū)﹂L(zhǎng)短鏈接的處理及運(yùn)行參數(shù)的保存

    服務(wù)器端Go程序?qū)﹂L(zhǎng)短鏈接的處理及運(yùn)行參數(shù)的保存

    這篇文章主要介紹了服務(wù)器端Go程序?qū)﹂L(zhǎng)短鏈接的處理及運(yùn)行參數(shù)的保存,這里針對(duì)使用Go語(yǔ)言編寫(xiě)的Socket服務(wù)器進(jìn)行實(shí)例說(shuō)明,需要的朋友可以參考下
    2016-03-03
  • mac下安裝golang框架iris的方法

    mac下安裝golang框架iris的方法

    這篇文章主要介紹了mac下安裝golang框架iris的方法,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-11-11
  • golang jwt+token驗(yàn)證的實(shí)現(xiàn)

    golang jwt+token驗(yàn)證的實(shí)現(xiàn)

    這篇文章主要介紹了golang jwt+token驗(yàn)證的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-10-10
  • 基于golang時(shí)間轉(zhuǎn)換的問(wèn)題

    基于golang時(shí)間轉(zhuǎn)換的問(wèn)題

    下面小編就為大家?guī)?lái)一篇基于golang時(shí)間轉(zhuǎn)換的問(wèn)題。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-08-08
  • 基于golang如何實(shí)現(xiàn)error工具包詳解

    基于golang如何實(shí)現(xiàn)error工具包詳解

    Go 語(yǔ)言使用 error 類型來(lái)返回函數(shù)執(zhí)行過(guò)程中遇到的錯(cuò)誤,下面這篇文章主要給大家介紹了關(guān)于如何基于golang實(shí)現(xiàn)error工具包的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2018-09-09
  • Go獲取兩個(gè)時(shí)間點(diǎn)時(shí)間差的具體實(shí)現(xiàn)

    Go獲取兩個(gè)時(shí)間點(diǎn)時(shí)間差的具體實(shí)現(xiàn)

    本文主要介紹了Go獲取兩個(gè)時(shí)間點(diǎn)時(shí)間差的具體實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • GO語(yǔ)言make()分配用法實(shí)例

    GO語(yǔ)言make()分配用法實(shí)例

    這篇文章主要介紹了GO語(yǔ)言make()分配用法,實(shí)例分析了make()的功能及使用技巧,需要的朋友可以參考下
    2015-02-02
  • 詳解golang中模板的常用語(yǔ)法

    詳解golang中模板的常用語(yǔ)法

    這篇文章主要介紹了golang模板中的常用語(yǔ)法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-08-08
  • Golang中time.After的使用理解與釋放問(wèn)題

    Golang中time.After的使用理解與釋放問(wèn)題

    這篇文章主要給大家介紹了關(guān)于Golang中time.After的使用理解與釋放問(wèn)題,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2018-08-08
  • 詳解Golang time包中的結(jié)構(gòu)體time.Time

    詳解Golang time包中的結(jié)構(gòu)體time.Time

    在日常開(kāi)發(fā)過(guò)程中,會(huì)頻繁遇到對(duì)時(shí)間進(jìn)行操作的場(chǎng)景,使用 Golang 中的 time 包可以很方便地實(shí)現(xiàn)對(duì)時(shí)間的相關(guān)操作,本文先講解一下 time 包中的結(jié)構(gòu)體 time.Time,需要的朋友可以參考下
    2023-07-07

最新評(píng)論