亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Go?語言使用goroutine運(yùn)行閉包踩坑分析

 更新時(shí)間:2023年11月30日 11:42:04   作者:frank  
這篇文章主要介紹了Go?語言使用goroutine運(yùn)行閉包踩坑解決分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

介紹

在 Go 語言中,函數(shù)支持匿名函數(shù),閉包就是一種特殊的匿名函數(shù),它可以用于訪問函數(shù)體外部的變量。

需要注意的是,在 for ... range ... 中,使用 goroutine 執(zhí)行閉包時(shí),經(jīng)常會(huì)掉“坑”。

因?yàn)槟涿瘮?shù)可以訪問函數(shù)體外部的變量,而 for ... range ... 返回的 val 的值是引用的同一個(gè)內(nèi)存地址的數(shù)據(jù),所以匿名函數(shù)訪問的函數(shù)體外部的 val 值是循環(huán)中最后輸出的一個(gè)值。

“踩坑”示例代碼

func main() {
 done := make(chan bool)
 values := []string{"a", "b", "c"}
 for _, v := range values {
  go func() {
   fmt.Println(v)
   done <- true
  }()
 }
 // wait for all goroutines to complete before exiting
 for _ = range values {
  <-done
 }
}

輸出結(jié)果:

c

c

c

閱讀上面這段代碼,在 for ... range ... 中,使用 goroutine 執(zhí)行閉包,打印切片中的元素,實(shí)際輸出結(jié)果不是我們期望得到的輸出結(jié)果。

這是因?yàn)檠h(huán)的每次迭代都使用相同的變量 v 實(shí)例,因此每個(gè)閉包共享該單個(gè)變量。我們可以在示例代碼中簡(jiǎn)單修改,同時(shí)輸出變量 v 的內(nèi)存地址和值。

把 fmt.Println(v) 修改為 fmt.Printf("val=%s pointer=%p\n", v, &v)。

修改后的輸出結(jié)果:

val=c pointer=0xc000010200
val=c pointer=0xc000010200
val=c pointer=0xc000010200

我們可以在輸出結(jié)果中發(fā)現(xiàn),打印變量 v 的內(nèi)存地址都是 0xc000010200。

當(dāng)閉包運(yùn)行時(shí),它會(huì)在執(zhí)行 fmt.Println 時(shí)打印變量 v 的值,但 v 的值可能在 goroutine 啟動(dòng)后已被修改。感興趣的讀者朋友們可以使用 go vet 檢查。

怎么避免“踩坑”呢?

一種方法是將變量作為參數(shù)傳遞給閉包:

func main() {
 done := make(chan bool)
 values := []string{"a", "b", "c"}
 for _, v := range values {
  go func(param string) {
   // fmt.Println(v)
   fmt.Printf("val=%s pointer=%p\n", param, &param)
   done <- true
  }(v)
 }
 // wait for all goroutines to complete before exiting
 for _ = range values {
  <-done
 }
}

輸出結(jié)果:

val=c pointer=0xc000010200
val=a pointer=0xc00009a000
val=b pointer=0xc0000a4000

閱讀上面這段代碼,通過將變量 v 的值作為參數(shù)傳遞給閉包,然后,該值作為形參 param 的值,在函數(shù)體內(nèi)部被訪問。

另外一種方法是創(chuàng)建一個(gè)新變量

func main() {
 done := make(chan bool)
 values := []string{"a", "b", "c"}
 for _, v := range values {
  param := v
  go func() {
   // fmt.Println(v)
   fmt.Printf("val=%s pointer=%p\n", param, &param)
   done <- true
  }()
 }
 // wait for all goroutines to complete before exiting
 for _ = range values {
  <-done
 }
}

輸出結(jié)果:

val=c pointer=0xc000082200
val=a pointer=0xc0000821e0
val=b pointer=0xc0000821f0

通過輸出結(jié)果可以發(fā)現(xiàn),該種方式也可以達(dá)到我們期望的結(jié)果。

總結(jié)

本文我們介紹了在 for ... range ... 中,Go 語言在每次迭代時(shí),沒有定義一個(gè)新變量,導(dǎo)致使用 goroutine 運(yùn)行閉包時(shí),經(jīng)常會(huì)掉“坑”。

我們給出避免“踩坑”的兩種方法,其中,第二種方法更簡(jiǎn)單,更多關(guān)于Go goroutine運(yùn)行閉包的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Golang 使用Map實(shí)現(xiàn)去重與set的功能操作

    Golang 使用Map實(shí)現(xiàn)去重與set的功能操作

    這篇文章主要介紹了Golang 使用 Map 實(shí)現(xiàn)去重與 set 的功能操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • 最新評(píng)論