golang中defer的關(guān)鍵特性示例詳解
前言
大家都知道golang的defer關(guān)鍵字,它可以在函數(shù)返回前執(zhí)行一些操作,最常用的就是打開(kāi)一個(gè)資源(例如一個(gè)文件、數(shù)據(jù)庫(kù)連接等)時(shí)就用defer延遲關(guān)閉改資源,以免引起內(nèi)存泄漏。本文主要給大家介紹了關(guān)于golang中defer的關(guān)鍵特性,分享出來(lái)供大家參考學(xué)習(xí),下面話不多說(shuō),來(lái)一起看看詳細(xì)的介紹:
一、defer 的作用和執(zhí)行時(shí)機(jī)
go 的 defer 語(yǔ)句是用來(lái)延遲執(zhí)行函數(shù)的,而且延遲發(fā)生在調(diào)用函數(shù) return 之后,比如
func a() int { defer b() return 0 }
b 的執(zhí)行是發(fā)生在 return 0 之后,注意 defer 的語(yǔ)法,關(guān)鍵字 defer 之后是函數(shù)的調(diào)用。
二、defer 的重要用途一:清理釋放資源
由于 defer 的延遲特性,defer 常用在函數(shù)調(diào)用結(jié)束之后清理相關(guān)的資源,比如
f, _ := os.Open(filename) defer f.Close()
文件資源的釋放會(huì)在函數(shù)調(diào)用結(jié)束之后借助 defer 自動(dòng)執(zhí)行,不需要時(shí)刻記住哪里的資源需要釋放,打開(kāi)和釋放必須相對(duì)應(yīng)。
用一個(gè)例子深刻詮釋一下 defer 帶來(lái)的便利和簡(jiǎn)潔。
代碼的主要目的是打開(kāi)一個(gè)文件,然后復(fù)制內(nèi)容到另一個(gè)新的文件中,沒(méi)有 defer 時(shí)這樣寫:
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 { //1 return } written, err = io.Copy(dst, src) dst.Close() src.Close() return }
代碼在 #1 處返回之后,src 文件沒(méi)有執(zhí)行關(guān)閉操作,可能會(huì)導(dǎo)致資源不能正確釋放,改用 defer 實(shí)現(xiàn):
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) }
src 和 dst 都能及時(shí)清理和釋放,無(wú)論 return 在什么地方執(zhí)行。
鑒于 defer 的這種作用,defer 常用來(lái)釋放數(shù)據(jù)庫(kù)連接,文件打開(kāi)句柄等釋放資源的操作。
三、defer 的重要用途二:執(zhí)行 recover
被 defer 的函數(shù)在 return 之后執(zhí)行,這個(gè)時(shí)機(jī)點(diǎn)正好可以捕獲函數(shù)拋出的 panic,因而 defer 的另一個(gè)重要用途就是執(zhí)行 recover。
recover 只有在 defer 中使用才更有意義,如果在其他地方使用,由于 program 已經(jīng)調(diào)用結(jié)束而提前返回而無(wú)法有效捕捉錯(cuò)誤。
package main import ( "fmt" ) func main() { defer func() { if ok := recover(); ok != nil { fmt.Println("recover") } }() panic("error") }
記住 defer 要放在 panic 執(zhí)行之前。
四、多個(gè) defer 的執(zhí)行順序
defer 的作用就是把關(guān)鍵字之后的函數(shù)執(zhí)行壓入一個(gè)棧中延遲執(zhí)行,多個(gè) defer 的執(zhí)行順序是后進(jìn)先出 LIFO :
defer func() { fmt.Println("1") }() defer func() { fmt.Println("2") }() defer func() { fmt.Println("3") }()
輸出順序是 321。
這個(gè)特性可以對(duì)一個(gè) array 實(shí)現(xiàn)逆序操作。
五、被 deferred 函數(shù)的參數(shù)在 defer 時(shí)確定
這是 defer 的特點(diǎn),一個(gè)函數(shù)被 defer 時(shí),它的參數(shù)在 defer 時(shí)進(jìn)行計(jì)算確定,即使 defer 之后參數(shù)發(fā)生修改,對(duì)已經(jīng) defer 的函數(shù)沒(méi)有影響,什么意思?看例子:
func a() { i := 0 defer fmt.Println(i) i++ return }
a 執(zhí)行輸出的是 0 而不是 1,因?yàn)?defer 時(shí),i 的值是 0,此時(shí)被 defer 的函數(shù)參數(shù)已經(jīng)進(jìn)行執(zhí)行計(jì)算并確定了。
再看一個(gè)例子:
func calc(index string, a, b int) int { ret := a + b fmt.Println(index, a, b, ret) return ret } func main() { a := 1 b := 2 defer calc("1", a, calc("10", a, b)) a = 0 return }
執(zhí)行代碼輸出
10 1 2 3 1 1 3 4
defer 函數(shù)的參數(shù) 第三個(gè)參數(shù)在 defer 時(shí)就已經(jīng)計(jì)算完成并確定,第二個(gè)參數(shù) a 也是如此,無(wú)論之后 a 變量是否修改都不影響。
六、被 defer 的函數(shù)可以讀取和修改帶名稱的返回值
func c() (i int) { defer func() { i++ }() return 1 }
被 defer 的函數(shù)是在 return 之后執(zhí)行,可以修改帶名稱的返回值,上面的函數(shù) c 返回的是 2。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
參考資料
https://blog.golang.org/defer-panic-and-recover
- Golang巧用defer進(jìn)行錯(cuò)誤處理的方法
- golang中defer的使用規(guī)則詳解
- Golang學(xué)習(xí)筆記之延遲函數(shù)(defer)的使用小結(jié)
- 聊聊golang的defer的使用
- Golang之defer 延遲調(diào)用操作
- Golang 的defer執(zhí)行規(guī)則說(shuō)明
- 聊聊golang中多個(gè)defer的執(zhí)行順序
- 簡(jiǎn)單聊聊Golang中defer預(yù)計(jì)算參數(shù)
- Golang異常處理之defer,panic,recover的使用詳解
- Golang的關(guān)鍵字defer的使用方法
相關(guān)文章
k8s在go語(yǔ)言中的使用及client?初始化簡(jiǎn)介
這篇文章主要為大家介紹了k8s在go語(yǔ)言中的使用及client?初始化簡(jiǎn)介,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-04-04Go語(yǔ)言異步API設(shè)計(jì)的扇入扇出模式詳解
這篇文章主要為大家介紹了Go語(yǔ)言異步API設(shè)計(jì)的扇入扇出模式示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08Golang遠(yuǎn)程調(diào)用框架RPC的具體使用
Remote Procedure Call (RPC) 是一種使用TCP協(xié)議從另一個(gè)系統(tǒng)調(diào)用應(yīng)用程序功能執(zhí)行的方法。Go有原生支持RPC服務(wù)器實(shí)現(xiàn),本文通過(guò)簡(jiǎn)單實(shí)例介紹RPC的實(shí)現(xiàn)過(guò)程2022-12-12Go Asynq異步任務(wù)處理的實(shí)現(xiàn)
Asynq是一個(gè)新興的異步任務(wù)處理解決方案,它提供了輕量級(jí)的、易于使用的API,本文主要介紹了Go Asynq異步任務(wù)處理的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2023-06-06go語(yǔ)言LeetCode題解720詞典中最長(zhǎng)的單詞
這篇文章主要為大家介紹了go語(yǔ)言LeetCode題解720詞典中最長(zhǎng)的單詞,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12