- 資訊首頁(yè) > 網(wǎng)絡(luò )安全 >
- golang中如何使用defer
本篇文章給大家分享的是有關(guān)golang中如何使用defer,小編覺(jué)得挺實(shí)用的,因此分享給大家學(xué)習,希望大家閱讀完這篇文章后可以有所收獲,話(huà)不多說(shuō),跟著(zhù)小編一起來(lái)看看吧。
在golang當中,defer代碼塊會(huì )在函數調用鏈表中增加一個(gè)函數調用。這個(gè)函數調用不是普通的函數調用,而是會(huì )在函數正常返回,也就是return之后添加一個(gè)函數調用。因此,defer通常用來(lái)釋放函數內部變量。
為了更好的學(xué)習defer的行為,我們首先來(lái)看下面一段代碼:
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}
這段代碼可以運行,但存在'安全隱患'。如果調用dst, err := os.Create(dstName)失敗,則函數會(huì )執行return退出運行。但之前創(chuàng )建的src(文件句柄)沒(méi)有被釋放。 上面這段代碼很簡(jiǎn)單,所以我們可以一眼看出存在文件未被釋放的問(wèn)題。 如果我們的邏輯復雜或者代碼調用過(guò)多時(shí),這樣的錯誤未必會(huì )被及時(shí)發(fā)現。 而使用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) }
通過(guò)defer,我們可以在代碼中優(yōu)雅的關(guān)閉/清理代碼中所使用的變量。defer作為golang清理變量的特性,有其獨有且明確的行為。以下是defer三條使用規則。
我們通過(guò)以下代碼來(lái)解釋這條規則:
func a() {i := 0defer fmt.Println(i) i++ return }
上面我們說(shuō)過(guò),defer函數會(huì )在return之后被調用。那么這段函數執行完之后,是不用應該輸出1呢?
讀者自行編譯看一下,結果輸出的是0. why?
這是因為雖然我們在defer后面定義的是一個(gè)帶變量的函數: fmt.Println(i). 但這個(gè)變量(i)在defer被聲明的時(shí)候,就已經(jīng)確定其確定的值了。 換言之,上面的代碼等同于下面的代碼:
func a() { i := 0defer fmt.Println(0) //因為i=0,所以此時(shí)就明確告訴golang在程序退出時(shí),執行輸出0的操作i++return}
為了更為明確的說(shuō)明這個(gè)問(wèn)題,我們繼續定義一個(gè)defer:
func a() { i := 0defer fmt.Println(i) //輸出0,因為i此時(shí)就是0i++defer fmt.Println(i) //輸出1,因為i此時(shí)就是1return}
通過(guò)運行結果,可以看到defer輸出的值,就是定義時(shí)的值。而不是defer真正執行時(shí)的變量值(很重要,搞不清楚的話(huà)就會(huì )產(chǎn)生于預期不一致的結果)
但為什么是先輸出1,在輸出0呢? 看下面的規則二。
當同時(shí)定義了多個(gè)defer代碼塊時(shí),golang安裝先定義后執行的順序依次調用defer。不要為什么,golang就是這么定義的。我們用下面的代碼加深記憶和理解:
func b() {for i := 0; i < 4; i++ {defer fmt.Print(i) } }
在循環(huán)中,依次定義了四個(gè)defer代碼塊。結合規則一,我們可以明確得知每個(gè)defer代碼塊應該輸出什么值。 安裝先進(jìn)后出的原則,我們可以看到依次輸出了3210.
先看下面的代碼:
func c() (i int) {defer func() { i++ }()return 1}
輸出結果是12. 在開(kāi)頭的時(shí)候,我們說(shuō)過(guò)defer是在return調用之后才執行的。 這里需要明確的是defer代碼塊的作用域仍然在函數之內,結合上面的函數也就是說(shuō),defer的作用域仍然在c函數之內。因此defer仍然可以讀取c函數內的變量(如果無(wú)法讀取函數內變量,那又如何進(jìn)行變量清除呢....)。
當執行return 1 之后,i的值就是1. 此時(shí)此刻,defer代碼塊開(kāi)始執行,對i進(jìn)行自增操作。 因此輸出2.
掌握了defer以上三條使用規則,那么當我們遇到defer代碼塊時(shí),就可以明確得知defer的預期結果。
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng )、來(lái)自互聯(lián)網(wǎng)轉載和分享為主,文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權請聯(lián)系QQ:712375056 進(jìn)行舉報,并提供相關(guān)證據,一經(jīng)查實(shí),將立刻刪除涉嫌侵權內容。
Copyright ? 2009-2021 56dr.com. All Rights Reserved. 特網(wǎng)科技 特網(wǎng)云 版權所有 珠海市特網(wǎng)科技有限公司 粵ICP備16109289號
域名注冊服務(wù)機構:阿里云計算有限公司(萬(wàn)網(wǎng)) 域名服務(wù)機構:煙臺帝思普網(wǎng)絡(luò )科技有限公司(DNSPod) CDN服務(wù):阿里云計算有限公司 中國互聯(lián)網(wǎng)舉報中心 增值電信業(yè)務(wù)經(jīng)營(yíng)許可證B2
建議您使用Chrome、Firefox、Edge、IE10及以上版本和360等主流瀏覽器瀏覽本網(wǎng)站