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

Golang 語言中 Context 的使用方式

開發(fā) 前端
在 Golang 語言并發(fā)編程中,經(jīng)常會遇到監(jiān)控 goroutine 運(yùn)行結(jié)束的場景,通常我們會想到使用 WaitGroup 和 chan + select,其中 WaitGroup 用于監(jiān)控一組 goroutine 是否全部運(yùn)行結(jié)束,chan + select 用于監(jiān)控一個 goroutine 是否運(yùn)行結(jié)束(取消一個 goroutine)。

01、介紹

在 Golang 語言并發(fā)編程中,經(jīng)常會遇到監(jiān)控 goroutine 運(yùn)行結(jié)束的場景,通常我們會想到使用 WaitGroup 和 chan + select,其中 WaitGroup 用于監(jiān)控一組 goroutine 是否全部運(yùn)行結(jié)束,chan + select 用于監(jiān)控一個 goroutine 是否運(yùn)行結(jié)束(取消一個 goroutine)。

如果我們需要監(jiān)控多個 goroutine 是否運(yùn)行結(jié)束(取消多個 goroutine),通常會使用 context,當(dāng)然使用 context 也可以用于監(jiān)控一個 goroutine 是否運(yùn)行結(jié)束(取消一個 goroutine)。我們在之前的文章已經(jīng)介紹過 Golang 語言標(biāo)準(zhǔn)庫 Context,未閱讀的讀者朋友們可以按需翻閱。本文我們主要介紹 Context 的一些使用方式。

02、取消一個 goroutine

使用 context 取消一個 goroutine,比較類似于使用 chan + select 的方式取消一個 goroutine。

示例代碼:

  1. func main () { 
  2.  ctx, cancel := context.WithCancel(context.Background()) 
  3.  go func(ctx context.Context) { 
  4.   for { 
  5.    select { 
  6.     case <-ctx.Done(): 
  7.      fmt.Println("goroutine 已停止"
  8.      return 
  9.    default
  10.     fmt.Println("goroutine 正在運(yùn)行"
  11.     time.Sleep(time.Second
  12.    } 
  13.   } 
  14.  }(ctx) 
  15.  time.Sleep(time.Second * 5) 
  16.  cancel() 
  17.  time.Sleep(time.Second * 5) 
  18.  fmt.Println("main goroutine 已結(jié)束"

閱讀上面這段代碼,我們首先使用 context.Background() 創(chuàng)建一個 context 樹的根節(jié)點(diǎn),然后使用 context.WithCancel() 創(chuàng)建一個可取消的子 context 類型的變量 ctx,作為參數(shù)傳遞給子 goroutine,用作跟蹤子 goroutine。

然后在子 goroutine 中,使用 for select 監(jiān)控 <-ctx.Done() 判斷子 goroutine 是否運(yùn)行結(jié)束。

最后使用 context.WithCancel() 返回的第二個值 CancelFunc 類型的 cancel 變量給子 goroutine 發(fā)送取消指令。

03、取消多個 goroutine

接下來,我們再來看一個使用 context 停止多個 goroutine 的示例。

  1. func main () { 
  2.  ctx, cancel := context.WithCancel(context.Background()) 
  3.   // 停止多個 goroutine 
  4.  go worker(ctx, "節(jié)點(diǎn)一"
  5.  go worker(ctx, "節(jié)點(diǎn)二"
  6.  go worker(ctx, "節(jié)點(diǎn)三"
  7.  time.Sleep(time.Second * 5) 
  8.  cancel() 
  9.  time.Sleep(time.Second * 5) 
  10.  fmt.Println("main goroutine 已結(jié)束"
  11.  
  12. func worker (ctx context.Context, node string) { 
  13.  for { 
  14.   select { 
  15.    case <-ctx.Done(): 
  16.     fmt.Println(node, "goroutine 已停止"
  17.     return 
  18.   default
  19.    fmt.Println(node, "goroutine 正在運(yùn)行"
  20.    time.Sleep(time.Second
  21.   } 
  22.  } 

閱讀上面這段代碼,我們使用 go 關(guān)鍵字啟動三個 worker goroutine,和上個示例一樣,首先創(chuàng)建一個 context 樹的根節(jié)點(diǎn),使用第一個返回值 context 類型的子 ctx 跟蹤每一個 worker goroutine,在 worker 中使用 for seclect 監(jiān)控 <-ctx.Done() 判斷子 goroutine 是否運(yùn)行結(jié)束,最后通過調(diào)用第二個返回值 CancelFunc 類型的 cancel 給子 goroutine 發(fā)送取消指令,此時所有子 context 都會接收到取消指令,goroutine 結(jié)束運(yùn)行。

04、上下文信息傳遞

我們在前面的示例中使用 WithCancel 函數(shù),用作取消 context,除此之外,可用作取消 Context 的函數(shù)還有 WithDeadline 函數(shù)和 WithTimeout 函數(shù),分別用于定時取消和超時取消,限于篇幅,本文不再贅述,感興趣的讀者可以查閱官方標(biāo)準(zhǔn)庫文檔。除了上述三個函數(shù)外,還有一個 WithValue 函數(shù),它是用作上下文信息傳遞的一個函數(shù)。

在 Golang 語言中,Context 包還有一個重要的作用,就是用作上下文信息傳遞,接下來我們介紹一下如何使用 WithValue 函數(shù)傳遞上下文信息。

示例代碼:

  1. func main () { 
  2.  ctx, cancel := context.WithCancel(context.Background()) 
  3.  // 傳遞上下文信息 
  4.  ctxValue := context.WithValue(ctx, "uid", 1) 
  5.  go func(ctx context.Context) { 
  6.   for { 
  7.    select { 
  8.     case <-ctx.Done(): 
  9.      fmt.Println(ctx.Value("uid"), "goroutine 已停止"
  10.      return 
  11.    default
  12.     fmt.Println("goroutine 正在運(yùn)行"
  13.     time.Sleep(time.Second
  14.    } 
  15.   } 
  16.  }(ctxValue) 
  17.  time.Sleep(time.Second * 5) 
  18.  cancel() 
  19.  time.Sleep(time.Second * 5) 
  20.  fmt.Println("main goroutine 已結(jié)束"

閱讀上面這段代碼,我們使用 WithValue 函數(shù)給子 goroutine 傳遞上下文信息 uid。WithValue 函數(shù)接收三個參數(shù),分別是 parent context,key 和 value。返回值是一個 context,我們可以在子 goroutine 中調(diào)用 Value 方法獲取傳遞的上下文信息。

05、總結(jié)

本文我們簡述了監(jiān)控 goroutine 的幾種方式,分別是 WaitGroup,chan + select 和 context。重點(diǎn)介紹了 context 的一些使用方式,分別是取消一個 goroutine,取消多個 goroutine 和傳遞上下文信息。關(guān)于定時取消和超時取消,感興趣的讀者可以參閱官方標(biāo)準(zhǔn)庫文檔。

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

2021-06-07 23:19:44

Golang語言 Defer

2023-11-13 21:55:12

Go編程

2024-06-05 14:35:26

2021-06-29 23:40:19

Golang語言并發(fā)

2022-02-13 23:10:46

Golang語言變量

2021-05-07 15:28:03

Kafka客戶端Sarama

2009-11-23 11:25:54

php curl

2019-08-28 09:04:02

Go語言Python操作系統(tǒng)

2011-05-13 17:25:34

C

2021-09-13 05:02:49

GogRPC語言

2024-05-10 08:36:40

Go語言對象

2023-05-15 08:50:58

ContextGolang

2020-08-12 08:51:19

Go語言Concurrency后臺

2011-05-25 13:22:05

PHPJSON

2024-04-01 00:02:56

Go語言代碼

2014-04-09 09:32:24

Go并發(fā)

2021-07-12 05:05:59

Golang語言字段

2021-01-29 08:56:13

Golang標(biāo)準(zhǔn)庫函數(shù)

2013-06-25 09:52:32

GoGo語言Go編程

2023-10-09 07:14:42

panicGo語言
點(diǎn)贊
收藏

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