golang中如何使用defer-創(chuàng)新互聯(lián)

本篇文章給大家分享的是有關(guān)golang中如何使用defer,小編覺得挺實用的,因此分享給大家學習,希望大家閱讀完這篇文章后可以有所收獲,話不多說,跟著小編一起來看看吧。

站在用戶的角度思考問題,與客戶深入溝通,找到瑪納斯網(wǎng)站設計與瑪納斯網(wǎng)站推廣的解決方案,憑借多年的經(jīng)驗,讓設計與互聯(lián)網(wǎng)技術(shù)結(jié)合,創(chuàng)造個性化、用戶體驗好的作品,建站類型包括:網(wǎng)站制作、成都做網(wǎng)站、企業(yè)官網(wǎng)、英文網(wǎng)站、手機端網(wǎng)站、網(wǎng)站推廣、空間域名、網(wǎng)頁空間、企業(yè)郵箱。業(yè)務覆蓋瑪納斯地區(qū)。

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

為了更好的學習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(文件句柄)沒有被釋放。 上面這段代碼很簡單,所以我們可以一眼看出存在文件未被釋放的問題。 如果我們的邏輯復雜或者代碼調(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ī)則一 當defer被聲明時,其參數(shù)就會被實時解析

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

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

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

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

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

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

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

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

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

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

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

當同時定義了多個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代碼塊應該輸出什么值。 安裝先進后出的原則,我們可以看到依次輸出了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)變量,那又如何進行變量清除呢....)。

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

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

以上就是golang中如何使用defer,小編相信有部分知識點可能是我們?nèi)粘9ぷ鲿姷交蛴玫降?。希望你能通過這篇文章學到更多知識。更多詳情敬請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。

另外有需要云服務器可以了解下創(chuàng)新互聯(lián)scvps.cn,海內(nèi)外云服務器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務器、裸金屬服務器、高防服務器、香港服務器、美國服務器、虛擬主機、免備案服務器”等云主機租用服務以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應用場景需求。

分享標題:golang中如何使用defer-創(chuàng)新互聯(lián)
URL分享:http://www.muchs.cn/article36/dpgjsg.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供品牌網(wǎng)站制作自適應網(wǎng)站、網(wǎng)站收錄、App設計、定制開發(fā)、網(wǎng)站排名

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

外貿(mào)網(wǎng)站制作