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

溫馨提示×

溫馨提示×

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

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

golang中defer的關鍵特性是什么

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

今天就跟大家聊聊有關golang中defer的關鍵特性是什么,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內容,希望大家根據這篇文章可以有所收獲。

一、defer 的作用和執行時機

go 的 defer 語句是用來延遲執行函數的,而且延遲發生在調用函數 return 之后,比如

func a() int {
 defer b()
 return 0
}

b 的執行是發生在 return 0 之后,注意 defer 的語法,關鍵字 defer 之后是函數的調用。

二、defer 的重要用途一:清理釋放資源

由于 defer 的延遲特性,defer 常用在函數調用結束之后清理相關的資源,比如

f, _ := os.Open(filename)
defer f.Close()

文件資源的釋放會在函數調用結束之后借助 defer 自動執行,不需要時刻記住哪里的資源需要釋放,打開和釋放必須相對應。

用一個例子深刻詮釋一下 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 { //1
  return
 }
 written, err = io.Copy(dst, src)
 dst.Close()
 src.Close()
 return
}

代碼在 #1 處返回之后,src 文件沒有執行關閉操作,可能會導致資源不能正確釋放,改用 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)
}

src 和 dst 都能及時清理和釋放,無論 return 在什么地方執行。

鑒于 defer 的這種作用,defer 常用來釋放數據庫連接,文件打開句柄等釋放資源的操作。

三、defer 的重要用途二:執行 recover

被 defer 的函數在 return 之后執行,這個時機點正好可以捕獲函數拋出的 panic,因而 defer 的另一個重要用途就是執行 recover。

recover 只有在 defer 中使用才更有意義,如果在其他地方使用,由于 program 已經調用結束而提前返回而無法有效捕捉錯誤。

package main
import (
 "fmt"
)
func main() {
 defer func() {
  if ok := recover(); ok != nil {
   fmt.Println("recover")
  }
 }()
 panic("error")
}

記住 defer 要放在 panic 執行之前。

四、多個 defer 的執行順序

defer 的作用就是把關鍵字之后的函數執行壓入一個棧中延遲執行,多個 defer 的執行順序是后進先出 LIFO :

defer func() { fmt.Println("1") }()
defer func() { fmt.Println("2") }()
defer func() { fmt.Println("3") }()

輸出順序是 321。

這個特性可以對一個 array 實現逆序操作。

五、被 deferred 函數的參數在 defer 時確定

這是 defer 的特點,一個函數被 defer 時,它的參數在 defer 時進行計算確定,即使 defer 之后參數發生修改,對已經 defer 的函數沒有影響,什么意思?看例子:

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

a 執行輸出的是 0 而不是 1,因為 defer 時,i 的值是 0,此時被 defer 的函數參數已經進行執行計算并確定了。

再看一個例子:

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
}

執行代碼輸出

10 1 2 3 
1 1 3 4

defer 函數的參數 第三個參數在 defer 時就已經計算完成并確定,第二個參數 a 也是如此,無論之后 a 變量是否修改都不影響。

六、被 defer 的函數可以讀取和修改帶名稱的返回值

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

被 defer 的函數是在 return 之后執行,可以修改帶名稱的返回值,上面的函數 c 返回的是 2。

看完上述內容,你們對golang中defer的關鍵特性是什么有進一步的了解嗎?如果還想了解更多知識或者相關內容,請關注億速云行業資訊頻道,感謝大家的支持。

向AI問一下細節

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

AI

江津市| 吉安市| 黑水县| 稷山县| 鄢陵县| 清徐县| 翁牛特旗| 宜兴市| 油尖旺区| 库车县| 天台县| 北海市| 廉江市| 手游| 武冈市| 龙泉市| 常宁市| 饶阳县| 塔河县| 广东省| 古浪县| 苗栗市| 齐齐哈尔市| 习水县| 茂名市| 建始县| 山东省| 高唐县| 惠安县| 武夷山市| 泊头市| 大同县| 邛崃市| 桐城市| 阿鲁科尔沁旗| 灵川县| 铜鼓县| 侯马市| 乌苏市| 郁南县| 涞水县|