主頁(yè) > 知識(shí)庫(kù) > 總結(jié)Go語(yǔ)言中defer的使用和注意要點(diǎn)

總結(jié)Go語(yǔ)言中defer的使用和注意要點(diǎn)

熱門(mén)標(biāo)簽:騰訊外呼系統(tǒng)價(jià)格 浙江人工智能外呼管理系統(tǒng) 最短的地圖標(biāo)注 百度地圖標(biāo)注搜索關(guān)鍵詞 成都呼叫中心外呼系統(tǒng)平臺(tái) 電銷(xiāo)機(jī)器人可以補(bǔ)救房產(chǎn)中介嗎 谷歌便利店地圖標(biāo)注 電梯外呼訪客系統(tǒng) ?兓?

前言

defer是golang語(yǔ)言中的關(guān)鍵字,用于資源的釋放,會(huì)在函數(shù)返回之前進(jìn)行調(diào)用。

一般采用如下模式:

f,err := os.Open(filename)
if err != nil {
  panic(err)
}
defer f.Close()

如果有多個(gè)defer表達(dá)式,調(diào)用順序類(lèi)似于棧,越后面的defer表達(dá)式越先被調(diào)用。

延時(shí)調(diào)用函數(shù)的語(yǔ)法如下:

defer func_name(param-list)

當(dāng)一個(gè)函數(shù)調(diào)用前有關(guān)鍵字 defer 時(shí), 那么這個(gè)函數(shù)的執(zhí)行會(huì)推遲到包含這個(gè) defer 語(yǔ)句的函數(shù)即將返回前才執(zhí)行. 例如:

func main() {
  defer fmt.Println("Fourth")
  fmt.Println("First")
  fmt.Println("Third")
}

最后打印順序如下:

First
Second
Third

需要注意的是, defer 調(diào)用的函數(shù)參數(shù)的值 defer 被定義時(shí)就確定了.

例如:

i := 1
defer fmt.Println("Deferred print:", i)
i++
fmt.Println("Normal print:", i)

打印的內(nèi)容如下:

Normal print: 2
Deferred print: 1

因此我們知道, 在 "defer fmt.Println("Deferred print:", i)" 調(diào)用時(shí), i 的值已經(jīng)確定了, 因此相當(dāng)于 defer fmt.Println("Deferred print:", 1) 了.

需要強(qiáng)調(diào)的時(shí), defer 調(diào)用的函數(shù)參數(shù)的值在 defer 定義時(shí)就確定了, 而 defer 函數(shù)內(nèi)部所使用的變量的值需要在這個(gè)函數(shù)運(yùn)行時(shí)才確定.

例如:

func f1() (r int) {
  r = 1
  defer func() {
    r++
    fmt.Println(r)
  }()
  r = 2
  return
}

func main() {
  f1()
}

上面的例子中, 最終打印的內(nèi)容是 "3", 這是因?yàn)樵?"r = 2" 賦值之后, 執(zhí)行了 defer 函數(shù), 因此在這個(gè)函數(shù)內(nèi), r 的值是2了, 自增后變?yōu)?.

defer 順序

如果有多個(gè)defer 調(diào)用, 則調(diào)用的順序是先進(jìn)后出的順序, 類(lèi)似于入棧出棧一樣:

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

最先執(zhí)行的是 "fmt.Println(4)" , 接著是 "fmt.Println(3)" 依次類(lèi)推, 最后的輸出如下:

4
3
2
1

defer 注意要點(diǎn)

defer 函數(shù)調(diào)用的執(zhí)行時(shí)機(jī)是外層函數(shù)設(shè)置返回值之后, 并且在即將返回之前.

例如:

func f1() (r int) {
  defer func() {
    r++
  }()
  return 0
}
func main() {
  fmt.Println(f1())
}

上面 fmt.Println(f1()) 打印的是什么呢? 很多朋友可能會(huì)認(rèn)為打印的是0, 但是正確答案是 1. 這是為什么呢?

要弄明白這個(gè)問(wèn)題, 我們需要牢記兩點(diǎn)

     1、defer 函數(shù)調(diào)用的執(zhí)行時(shí)機(jī)是外層函數(shù)設(shè)置返回值之后, 并且在即將返回之前

     2、return XXX 操作并不是原子的.

我們將上面的例子改寫(xiě)一下大家就很明白了:

func f1() (r int) {
  defer func() {
    r++
  }()
  r = 0
  return
}

當(dāng)進(jìn)行賦值操作 "r = 0" 后, 才調(diào)用 defer 函數(shù), 最后才是返回語(yǔ)句.

因此上面的代碼等效于:

func f1() (r int) {
  r = 0
  func() {
    r++
  }()
  return
}

接下來(lái)我們?cè)賮?lái)看一個(gè)更有意思的例子:

func double(x int) int {
  return x + x
}

func triple(x int) (r int) {
  defer func() {
    r += x
  }()
  return double(x)
}

func main() {
  fmt.Println(triple(3))
}

如果我們已經(jīng)理解了上面所說(shuō)的內(nèi)容的話, 那么 triple 函數(shù)就很好理解了, 它實(shí)際上是:

func triple(x int) (r int) {
  r = double(x)
  func() {
    r += x
  }()
  return
}

defer 表達(dá)式的使用場(chǎng)景

defer 通常用于 open/close, connect/disconnect, lock/unlock 等這些成對(duì)的操作, 來(lái)保證在任何情況下資源都被正確釋放. 在這個(gè)角度來(lái)說(shuō), defer 操作和 Java 中的 try ... finally 語(yǔ)句塊有異曲同工之處.

例如:

var mutex sync.Mutex
var count = 0

func increment() {
  mutex.Lock()
  defer mutex.Unlock()
  count++
}

increment 函數(shù)中, 我們?yōu)榱吮苊飧?jìng)態(tài)條件的出現(xiàn), 而使用了 Mutex 進(jìn)行加鎖. 而在進(jìn)行并發(fā)編程時(shí), 加鎖了卻忘記(或某種情況下 unlock 沒(méi)有被執(zhí)行), 往往會(huì)造成災(zāi)難性的后果. 為了在任意情況下, 都要保證在加鎖操作后, 都進(jìn)行對(duì)應(yīng)的解鎖操作, 我們可以使用 defer 調(diào)用解鎖操作.

總結(jié)

以上就是這篇文章的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)或者工作帶來(lái)一定的幫助。如果有疑問(wèn)大家可以留言交流。

您可能感興趣的文章:
  • Go語(yǔ)言中的延遲函數(shù)defer示例詳解
  • Golang巧用defer進(jìn)行錯(cuò)誤處理的方法
  • GO語(yǔ)言延遲函數(shù)defer用法分析
  • C++實(shí)現(xiàn)Go的defer功能(示例代碼)

標(biāo)簽:盤(pán)錦 眉山 七臺(tái)河 紹興 邢臺(tái) 宜昌 上海 雅安

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《總結(jié)Go語(yǔ)言中defer的使用和注意要點(diǎn)》,本文關(guān)鍵詞  總結(jié),語(yǔ),言中,defer,的,使用,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《總結(jié)Go語(yǔ)言中defer的使用和注意要點(diǎn)》相關(guān)的同類(lèi)信息!
  • 本頁(yè)收集關(guān)于總結(jié)Go語(yǔ)言中defer的使用和注意要點(diǎn)的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章