溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

golang中defer的關(guān)鍵特性是什么

發(fā)布時(shí)間:2021-07-06 15:33:48 來(lái)源:億速云 閱讀:102 作者:Leah 欄目:編程語(yǔ)言

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

一、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í)這樣寫(xiě):

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。

看完上述內(nèi)容,你們對(duì)golang中defer的關(guān)鍵特性是什么有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI