成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一篇學會Go并發等待

開發 后端
關于 goroutine stack size(棧內存大小) 官方的文檔 中所述,1.2 之前最小是4kb,在1.2 變成8kb,并且可以使用SetMaxStack 設置棧最大大小。

[[411766]]

上節答疑

上一節有讀者問goroutine stack size一般是多大,我進行了詳細的查詢

關于 goroutine stack size(棧內存大小) 官方的文檔 中所述,1.2 之前最小是4kb,在1.2 變成8kb,并且可以使用SetMaxStack 設置棧最大大小。

在 runtime/debug 包能控制最大的單個 goroutine 的堆棧的大小。在 64 位系統上默認為 1GB,在 32 位系統上默認為 250MB。

因為每個goroutine需要能夠運行,所以它們都有自己的棧。假如每個goroutine分配固定棧大小并且不能增長,太小則會導致溢出,太大又會浪費空間,無法存在許多的goroutine。

所以在1.3版本中,改為了 Contiguous stack( 連續棧 ),為了解決這個問題,goroutine可以初始時只給棧分配很小的空間(8KB),然后隨著使用過程中的需要自動地增長。這就是為什么Go可以開千千萬萬個goroutine而不會耗盡內存。

1.4 版本 goroutine 堆棧從 8Kb 減少到 2Kb

Golang并發等待

本節源碼位置 https://github.com/golang-minibear2333/golang/blob/master/4.concurrent/goroutine-wait/”

簡介

goroutine 是 Golang 中非常有用的功能,有時候 goroutine 沒執行完函數就返回了,如果希望等待當前的 goroutine 執行完成再接著往下執行,該怎么辦?

  1. func say(s string) { 
  2.     for i := 0; i < 3; i++ { 
  3.         time.Sleep(100 * time.Millisecond) 
  4.         fmt.Println(s) 
  5.     } 
  6.  
  7. func main() { 
  8.     go say("hello world"
  9.     fmt.Println("over!"

輸出 over! , 主線程沒有等待

使用 Sleep 等待

  1. func main() { 
  2.     go say("hello world"
  3.     time.Sleep(time.Second*1) 
  4.     fmt.Println("over!"

運行修改后的程序,結果如下:

  1. hello world 
  2. hello world 
  3. hello world 
  4. over! 

結果符合預期,但是太 low 了,我們不知道實際執行中應該等待多長時間,所以不能接受這個方案!

發送信號

  1. func main() { 
  2.     done := make(chan bool) 
  3.     go func() { 
  4.         for i := 0; i < 3; i++ { 
  5.             time.Sleep(100 * time.Millisecond) 
  6.             fmt.Println("hello world"
  7.         } 
  8.         done <- true 
  9.     }() 
  10.  
  11.     <-done 
  12.     fmt.Println("over!"

輸出的結果和上面相同,也符合預期

這種方式不能處理多個協程,所以也不是優雅的解決方式。

WaitGroup

Golang 官方在 sync 包中提供了 WaitGroup 類型可以解決這個問題。其文檔描述如下:

使用方法可以總結為下面幾點:

  • 在父協程中創建一個 WaitGroup 實例,比如名稱為:wg
  • 調用 wg.Add(n) ,其中 n 是等待的 goroutine 的數量
  • 在每個 goroutine 運行的函數中執行 defer wg.Done()
  • 調用 wg.Wait() 阻塞主邏輯
  • 直到所有 goroutine 執行完成。
  1. func main() { 
  2.     var wg sync.WaitGroup 
  3.     wg.Add(2) 
  4.     go say2("hello", &wg) 
  5.     go say2("world", &wg) 
  6.     fmt.Println("over!"
  7.     wg.Wait() 
  8.  
  9. func say2(s string, waitGroup *sync.WaitGroup) { 
  10.     defer waitGroup.Done() 
  11.  
  12.     for i := 0; i < 3; i++ { 
  13.         fmt.Println(s) 
  14.     } 

輸出,注意順序混亂是因為并發執行

  1. hello 
  2. hello 
  3. hello 
  4. over! 
  5. world 
  6. world 
  7. world 

小心缺陷

簡短的例子,注意循環傳入的變量用中間變量替代,防止閉包 bug

  1. func errFunc() { 
  2.  var wg sync.WaitGroup 
  3.  sList := []string{"a""b"
  4.  wg.Add(len(sList)) 
  5.  for _, d := range sList { 
  6.   go func() { 
  7.    defer wg.Done() 
  8.    fmt.Println(d) 
  9.   }() 
  10.  } 
  11.  wg.Wait() 

輸出,可以發現全部變成了最后一個

父協程與子協程是并發的。父協程上的for循環瞬間執行完了,內部的協程使用的是d最后的值,這就是閉包問題。

解決方法當作參數傳入

  1. func correctFunc() { 
  2.  var wg sync.WaitGroup 
  3.  sList := []string{"a""b"
  4.  wg.Add(len(sList)) 
  5.  for _, d := range sList { 
  6.   go func(str string) { 
  7.    defer wg.Done() 
  8.    fmt.Println(str) 
  9.   }(d) 
  10.  } 
  11.  wg.Wait() 

輸出

要留意 range 中的value有可能出現 1.7.3 有可能會遇到的坑!

責任編輯:武曉燕 來源: 機智的程序員小熊
相關推薦

2022-05-17 08:02:55

GoTryLock模式

2022-06-09 08:41:17

Go網絡庫Gnet

2021-06-24 06:35:00

Go語言進程

2021-03-24 06:06:13

Go并發編程Singlefligh

2023-12-05 07:14:27

AIGo

2021-11-15 10:29:39

Go語言類型

2024-05-10 08:15:32

go語言反射機制

2022-01-02 08:43:46

Python

2022-02-07 11:01:23

ZooKeeper

2021-07-02 09:45:29

MySQL InnoDB數據

2021-07-06 08:59:18

抽象工廠模式

2023-01-03 08:31:54

Spring讀取器配置

2023-11-28 08:29:31

Rust內存布局

2021-07-05 22:11:38

MySQL體系架構

2022-08-26 09:29:01

Kubernetes策略Master

2022-08-23 08:00:59

磁盤性能網絡

2021-05-11 08:54:59

建造者模式設計

2020-12-23 08:39:11

Go語言基礎技術

2021-09-28 08:59:30

復原IP地址

2021-10-14 10:22:19

逃逸JVM性能
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 在线看片网站 | 日韩欧美三区 | 91香蕉视频在线观看 | 亚洲一本| 久久99精品久久久久久琪琪 | 91人人视频在线观看 | 久久99深爱久久99精品 | 久久久蜜桃 | 久久亚洲国产精品 | 欧美日韩在线不卡 | 毛片在线免费播放 | 亚洲精品字幕 | 欧美成人a| 国产91网址 | 99精品久久 | 欧美成人精品一区二区男人看 | 国产中文字幕在线 | 一区二区三区福利视频 | 亚洲国产精品99久久久久久久久 | 国产精品毛片一区二区在线看 | 懂色中文一区二区三区在线视频 | 免费观看www7722午夜电影 | 国产精品99久久久久久久久 | 日韩第一夜 | 超碰一区二区 | 午夜欧美a级理论片915影院 | 精品一区二区三 | 国内精品久久精品 | 国产精品一区在线观看 | 91久久久久久久久 | 日韩一区二区在线视频 | 在线观看中文字幕视频 | 一区二区三区视频免费观看 | 黄色亚洲 | 中文字幕在线精品 | 日韩在线精品视频 | 九色在线观看 | 超碰欧美 | 一级黄色片网站 | 欧美日本在线 | 91天堂 |