中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

golang中defer如何使用

發布時間:2021-07-06 15:33:22 來源:億速云 閱讀:135 作者:Leah 欄目:編程語言

golang中defer如何使用,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

前言

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

為了更好的學習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 
}

這段代碼可以運行,但存在'安全隱患'。如果調用dst, err := os.Create(dstName)失敗,則函數會執行return退出運行。但之前創建的src(文件句柄)沒有被釋放。 上面這段代碼很簡單,所以我們可以一眼看出存在文件未被釋放的問題。 如果我們的邏輯復雜或者代碼調用過多時,這樣的錯誤未必會被及時發現。 而使用defer則可以避免這種情況的發生,下面是使用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,我們可以在代碼中優雅的關閉/清理代碼中所使用的變量。defer作為golang清理變量的特性,有其獨有且明確的行為。以下是defer三條使用規則。

規則一 當defer被聲明時,其參數就會被實時解析

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

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

上面我們說過,defer函數會在return之后被調用。那么這段函數執行完之后,是不用應該輸出1呢?

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

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

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

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

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

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

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

規則二 defer執行順序為先進后出

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

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

在循環中,依次定義了四個defer代碼塊。結合規則一,我們可以明確得知每個defer代碼塊應該輸出什么值。 安裝先進后出的原則,我們可以看到依次輸出了3210.

規則三 defer可以讀取有名返回值

先看下面的代碼:

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

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

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

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

看完上述內容是否對您有幫助呢?如果還想對相關知識有進一步的了解或閱讀更多相關文章,請關注億速云行業資訊頻道,感謝您對億速云的支持。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

绵竹市| 奉贤区| 包头市| 博兴县| 万山特区| 密云县| 千阳县| 中宁县| 治多县| 西华县| 西乌珠穆沁旗| 峨山| 镇原县| 凤阳县| 马关县| 克什克腾旗| 尼玛县| 额济纳旗| 泸西县| 九龙城区| 新蔡县| 石河子市| 五华县| 越西县| 和田县| 综艺| 祁阳县| 安丘市| 玛纳斯县| 青海省| 涞源县| 正宁县| 镇原县| 洪湖市| 麻栗坡县| 灵山县| 兴和县| 江源县| 六枝特区| 佛山市| 龙井市|