主頁 > 知識庫 > golang中defer的使用規(guī)則詳解

golang中defer的使用規(guī)則詳解

熱門標(biāo)簽:釘釘有地圖標(biāo)注功能嗎 汕頭小型外呼系統(tǒng) 阿里云ai電話機器人 惠州電銷防封電話卡 浙江高頻外呼系統(tǒng)多少錢一個月 黃岡人工智能電銷機器人哪個好 建造者2地圖標(biāo)注 濱州自動電銷機器人排名 鄭州亮點科技用的什么外呼系統(tǒng)

前言

在golang當(dāng)中,defer代碼塊會在函數(shù)調(diào)用鏈表中增加一個函數(shù)調(diào)用。這個函數(shù)調(diào)用不是普通的函數(shù)調(diào)用,而是會在函數(shù)正常返回,也就是return之后添加一個函數(shù)調(diào)用。因此,defer通常用來釋放函數(shù)內(nèi)部變量。

為了更好的學(xué)習(xí)defer的行為,我們首先來看下面一段代碼:

func CopyFile(dstName, srcName string) (written int64, err error) { 
src, err := os.Open(srcName) 
if err != nil { 
return 
}

dst, err := os.Create(dstName) 
if err != nil { 
return 
}

written, err = io.Copy(dst, src) 
dst.Close() 
src.Close() 
return 
}

這段代碼可以運行,但存在'安全隱患'。如果調(diào)用dst, err := os.Create(dstName)失敗,則函數(shù)會執(zhí)行return退出運行。但之前創(chuàng)建的src(文件句柄)沒有被釋放。 上面這段代碼很簡單,所以我們可以一眼看出存在文件未被釋放的問題。 如果我們的邏輯復(fù)雜或者代碼調(diào)用過多時,這樣的錯誤未必會被及時發(fā)現(xiàn)。 而使用defer則可以避免這種情況的發(fā)生,下面是使用defer的代碼:

func CopyFile(dstName, srcName string) (written int64, err error) { 
src, err := os.Open(srcName) 
if err != nil { 
return 
}
defer src.Close()

dst, err := os.Create(dstName) 
if err != nil { 
return 
}
defer dst.Close()

return io.Copy(dst, src) 
}

通過defer,我們可以在代碼中優(yōu)雅的關(guān)閉/清理代碼中所使用的變量。defer作為golang清理變量的特性,有其獨有且明確的行為。以下是defer三條使用規(guī)則。

規(guī)則一 當(dāng)defer被聲明時,其參數(shù)就會被實時解析

我們通過以下代碼來解釋這條規(guī)則:

func a() { 
i := 0 
defer fmt.Println(i) 
i++ 
return 
}

上面我們說過,defer函數(shù)會在return之后被調(diào)用。那么這段函數(shù)執(zhí)行完之后,是不用應(yīng)該輸出1呢?

讀者自行編譯看一下,結(jié)果輸出的是0. why?

這是因為雖然我們在defer后面定義的是一個帶變量的函數(shù): fmt.Println(i) . 但這個變量(i)在defer被聲明的時候,就已經(jīng)確定其確定的值了。 換言之,上面的代碼等同于下面的代碼:

func a() { 
i := 0 
defer fmt.Println(0) //因為i=0,所以此時就明確告訴golang在程序退出時,執(zhí)行輸出0的操作 
i++ 
return 
}

為了更為明確的說明這個問題,我們繼續(xù)定義一個defer:

func a() { 
i := 0 
defer fmt.Println(i) //輸出0,因為i此時就是0 
i++ 
defer fmt.Println(i) //輸出1,因為i此時就是1 
return 
}

通過運行結(jié)果,可以看到defer輸出的值,就是定義時的值。而不是defer真正執(zhí)行時的變量值(很重要,搞不清楚的話就會產(chǎn)生于預(yù)期不一致的結(jié)果)

但為什么是先輸出1,在輸出0呢? 看下面的規(guī)則二。

規(guī)則二 defer執(zhí)行順序為先進(jìn)后出

當(dāng)同時定義了多個defer代碼塊時,golang安裝先定義后執(zhí)行的順序依次調(diào)用defer。不要為什么,golang就是這么定義的。我們用下面的代碼加深記憶和理解:

func b() { 
for i := 0; i  4; i++ { 
defer fmt.Print(i) 
}
}

在循環(huán)中,依次定義了四個defer代碼塊。結(jié)合規(guī)則一,我們可以明確得知每個defer代碼塊應(yīng)該輸出什么值。 安裝先進(jìn)后出的原則,我們可以看到依次輸出了3210.

規(guī)則三 defer可以讀取有名返回值

先看下面的代碼:

func c() (i int) { 
defer func() { i++ }() 
return 1 
}

輸出結(jié)果是12. 在開頭的時候,我們說過defer是在return調(diào)用之后才執(zhí)行的。 這里需要明確的是defer代碼塊的作用域仍然在函數(shù)之內(nèi),結(jié)合上面的函數(shù)也就是說,defer的作用域仍然在c函數(shù)之內(nèi)。因此defer仍然可以讀取c函數(shù)內(nèi)的變量(如果無法讀取函數(shù)內(nèi)變量,那又如何進(jìn)行變量清除呢....)。

當(dāng)執(zhí)行return 1 之后,i的值就是1. 此時此刻,defer代碼塊開始執(zhí)行,對i進(jìn)行自增操作。 因此輸出2.

掌握了defer以上三條使用規(guī)則,那么當(dāng)我們遇到defer代碼塊時,就可以明確得知defer的預(yù)期結(jié)果。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • Golang之defer 延遲調(diào)用操作
  • 聊聊golang的defer的使用
  • Golang學(xué)習(xí)筆記之延遲函數(shù)(defer)的使用小結(jié)
  • golang中defer的關(guān)鍵特性示例詳解
  • Golang巧用defer進(jìn)行錯誤處理的方法
  • Golang 的defer執(zhí)行規(guī)則說明

標(biāo)簽:東營 駐馬店 晉中 阿壩 瀘州 滄州 泰安 昭通

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《golang中defer的使用規(guī)則詳解》,本文關(guān)鍵詞  golang,中,defer,的,使用,規(guī)則,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《golang中defer的使用規(guī)則詳解》相關(guān)的同類信息!
  • 本頁收集關(guān)于golang中defer的使用規(guī)則詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章