自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Golang 語(yǔ)言怎么控制并發(fā) Goroutine?

開發(fā) 后端
Golang 語(yǔ)言的優(yōu)勢(shì)之一是天生支持并發(fā),我們?cè)?Golang 語(yǔ)言開發(fā)中,通常使用的并發(fā)控制方式主要有 Channel,WaitGroup 和 Context,本文我們主要介紹一下 Golang 語(yǔ)言中并發(fā)控制的這三種方式怎么使用?

 [[408188]]

01介紹

Golang 語(yǔ)言的優(yōu)勢(shì)之一是天生支持并發(fā),我們?cè)?Golang 語(yǔ)言開發(fā)中,通常使用的并發(fā)控制方式主要有 Channel,WaitGroup 和 Context,本文我們主要介紹一下 Golang 語(yǔ)言中并發(fā)控制的這三種方式怎么使用?關(guān)于它們各自的詳細(xì)介紹在之前的文章已經(jīng)介紹過,感興趣的讀者朋友們可以按需翻閱。

02Channel

在 Golang 語(yǔ)言中,Channel 不僅可以用于協(xié)程之間通信,還可以使用 Channel 控制子協(xié)程,而且使用 Channel 實(shí)現(xiàn)并發(fā)控制比較簡(jiǎn)單,比如以下示例,我們?cè)?Golang 應(yīng)用程序中啟動(dòng)兩個(gè)協(xié)程,分別是主協(xié)程和子協(xié)程,主協(xié)程需要等待子協(xié)程運(yùn)行結(jié)束后再退出程序。

示例代碼:

  1. func main () { 
  2.  done := make(chan struct{}) 
  3.  go func() { 
  4.   fmt.Println("goroutine run over"
  5.   done <- struct{}{} 
  6.  }() 
  7.  <- done 
  8.  fmt.Println("main goroutine run over"

閱讀上面這段代碼,我們?cè)谧?goroutine 運(yùn)行結(jié)束后,通過 Channel 通知主 goroutine 退出程序,實(shí)際上也可以反過來處理,主 goroutine 通知子 goroutine 退出程序,主 goroutine 向 channel 中發(fā)送數(shù)據(jù),子 goroutine 等待接收 channel 中的數(shù)據(jù)。

03sync.WaitGroup

如果在 Golang 應(yīng)用程序中,需要讓主 goroutine 等待多個(gè) goroutine 都運(yùn)行結(jié)束后再退出程序,我們應(yīng)該怎么實(shí)現(xiàn)呢?是的,同樣可以使用 Channel 實(shí)現(xiàn),但是,有一個(gè)更優(yōu)雅的實(shí)現(xiàn)方式,那就是 WaitGroup,顧名思義,WaitGroup 就是等待一組 goroutine 運(yùn)行結(jié)束。

示例代碼:

  1. func main () { 
  2.  wg := sync.WaitGroup{} 
  3.  wg.Add(10) 
  4.  for i := 0; i < 10; i++ { 
  5.   go func(id int) { 
  6.    fmt.Println(id, "運(yùn)行結(jié)束"
  7.    wg.Done() 
  8.   }(i) 
  9.  } 
  10.  wg.Wait() 
  11.  fmt.Println("main goroutine run over"

閱讀上面這段代碼,我們啟動(dòng) 10 個(gè)子 goroutine,主 goroutine 需要等待 10 個(gè)子 goroutine 都運(yùn)行結(jié)束后再退出程序,我們使用的是 WaitGroup,它有三個(gè)方法,分別是 Add、Done 和 Wait,實(shí)際上 WaitGroup 維護(hù)了一個(gè)計(jì)數(shù)器,這三個(gè)方法都是圍繞這個(gè)計(jì)數(shù)器工作,Add 用于設(shè)置計(jì)數(shù)器的數(shù)值,Done 用于扣減計(jì)數(shù)器的數(shù)值,Wait 在計(jì)數(shù)器數(shù)值為 0 之前一直阻塞。關(guān)于 WaitGroup 的源碼解讀,在之前的文章中已介紹過,限于篇幅,這里就不再贅述。

04Context

Channel 和 WaitGroup 通常用于父子兩個(gè)層級(jí)的 goroutine 的應(yīng)用程序的并發(fā)控制中,如果在 Golang 應(yīng)用程序中,子協(xié)程繼續(xù)派生出協(xié)程,我們應(yīng)該怎么控制呢?這種多級(jí) goroutine 的應(yīng)用程序,我們可以使用 Context 實(shí)現(xiàn)并發(fā)控制。

示例代碼:

  1. func main() { 
  2.  ctx, cancel := context.WithCancel(context.Background()) 
  3.  go firstCtx(ctx) 
  4.  time.Sleep(5 * time.Second
  5.  fmt.Println("stop all sub goroutine"
  6.  cancel() 
  7.  time.Sleep(5 * time.Second
  8.  
  9. func firstCtx(ctx context.Context) { 
  10.  go secondCtx(ctx) 
  11.  for { 
  12.   select { 
  13.   case <-ctx.Done(): 
  14.    fmt.Println("first done"
  15.    return 
  16.   default
  17.    fmt.Println("first running"
  18.    time.Sleep(2 * time.Second
  19.   } 
  20.  } 
  21.  
  22. func secondCtx(ctx context.Context) { 
  23.  for { 
  24.   select { 
  25.   case <-ctx.Done(): 
  26.    fmt.Println("second done"
  27.    return 
  28.   default
  29.    fmt.Println("second running"
  30.    time.Sleep(2 * time.Second
  31.   } 
  32.  } 

閱讀上面這段代碼,在子協(xié)程 firstCtx 啟動(dòng)子協(xié)程 secondCtx,主 goroutine 創(chuàng)建 context,并把 context 傳遞到所有子協(xié)程,然后主 goroutine 通過調(diào)用 cancle 停掉所有子協(xié)程。

05總結(jié)

本文我們介紹了不同場(chǎng)景中分別適合哪種控制并發(fā) goroutine 的方式,其中,channel 適合控制少量 并發(fā) goroutine,WaitGroup 適合控制一組并發(fā) goroutine,而 context 適合控制多級(jí)并發(fā) goroutine。

本文轉(zhuǎn)載自微信公眾號(hào)「Golang語(yǔ)言開發(fā)?!梗梢酝ㄟ^以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系Golang語(yǔ)言開發(fā)棧公眾號(hào)。

 

責(zé)任編輯:武曉燕 來源: Golang語(yǔ)言開發(fā)棧
相關(guān)推薦

2021-09-30 09:21:28

Go語(yǔ)言并發(fā)編程

2025-01-15 09:13:53

2021-09-13 05:02:49

GogRPC語(yǔ)言

2021-06-09 23:36:46

Golang語(yǔ)言版本

2023-01-30 15:41:10

Channel控制并發(fā)

2021-10-10 23:02:49

Golang語(yǔ)言代碼

2021-12-13 01:24:14

語(yǔ)言Golang panic

2021-04-28 09:02:48

Golang語(yǔ)言Context

2022-04-13 08:23:31

Golang并發(fā)

2021-11-08 23:09:07

Go排序數(shù)據(jù)

2021-06-07 23:19:44

Golang語(yǔ)言 Defer

2021-10-31 23:01:50

語(yǔ)言拼接字符串

2021-07-22 09:43:09

Golang語(yǔ)言并發(fā)機(jī)制

2023-05-22 09:27:11

GMPGolang

2023-08-21 07:34:37

GolangGMP

2017-08-21 10:56:55

MySQL并發(fā)控制

2022-08-08 08:31:55

Go 語(yǔ)言閉包匿名函數(shù)

2022-01-04 23:13:57

語(yǔ)言PanicGolang

2022-08-08 06:50:06

Go語(yǔ)言閉包

2021-07-12 05:05:59

Golang語(yǔ)言字段
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)