婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

主頁 > 知識庫 > Golang優雅關閉channel的方法示例

Golang優雅關閉channel的方法示例

熱門標簽:汕頭小型外呼系統 建造者2地圖標注 濱州自動電銷機器人排名 釘釘有地圖標注功能嗎 惠州電銷防封電話卡 黃岡人工智能電銷機器人哪個好 浙江高頻外呼系統多少錢一個月 阿里云ai電話機器人 鄭州亮點科技用的什么外呼系統

前言

最近使用go開發后端服務,服務關閉需要保證channel中的數據都被讀取完,理由很簡單,在收到系統的中斷信號后,系統需要做收尾工作,保證channel的數據都要被處理掉,然后才可以關閉系統。但實現起來沒那么簡單,下面來一起看看詳細的介紹吧。

關于Go channel設計和規范的批評:

  • 在不能更改channel狀態的情況下,沒有簡單普遍的方式來檢查channel是否已經關閉了
  • 關閉已經關閉的channel會導致panic,所以在closer(關閉者)不知道channel是否已經關閉的情況下去關閉channel是很危險的
  • 發送值到已經關閉的channel會導致panic,所以如果sender(發送者)在不知道channel是否已經關閉的情況下去向channel發送值是很危險的

所以Golang 內建的 close 方法可以關閉 channel,如果往已經關閉的 channel 發送數據,則會報錯:panic: close of closed channel.

看如下代碼,在一段時間內,生產者可以不斷往 channel 寫入數據,消費者進行處理,一段時間后 channel 關閉了,這個時候如果還有數據往 channel 發送,程序就會報錯。

package main
 
import (
 "fmt"
 "sync"
 "time"
)
 
func main() {
 jobs := make(chan int)
 var wg sync.WaitGroup
 go func() {
 time.Sleep(time.Second * 3)
 close(jobs)
 }()
 go func() {
 for i := 0; ; i++ {
 jobs - i
 fmt.Println("produce:", i)
 }
 }()
 wg.Add(1)
 go func() {
 defer wg.Done()
 for i := range jobs {
 fmt.Println("consume:", i)
 }
 }()
 wg.Wait()
}

多運行幾次出錯的概率會比較大:

produce: 33334
consume: 33334
consume: 33335
produce: 33335
produce: 33336
consume: 33336
consume: 33337
produce: 33337
produce: 33338
consume: 33338
consume: 33339
produce: 33339
produce: 33340
consume: 33340
panic: send on closed channel
 
goroutine 19 [running]:
panic(0x49b660, 0xc042410bb0)
  C:/Go/src/runtime/panic.go:500 +0x1af
main.main.func2(0xc04203a180)
  C:/Users/tanteng/Go/src/examples/channel_close.go:18 +0x6b
created by main.main
  C:/Users/tanteng/Go/src/examples/channel_close.go:21 +0xb8
exit status 2

如何優雅關閉 channel

那么在往通道發數據前如何判斷通道是否關閉呢?

1._,ok := - jobs

此時如果 channel 關閉,ok 值為 false,如果 channel 沒有關閉,則會漏掉一個 jobs

2.使用 select 方式

再創建一個 channel,叫做 timeout,如果超時往這個 channel 發送 true,在生產者發送數據給 jobs 的 channel,用 select 監聽 timeout,如果超時則關閉 jobs 的 channel.

完整代碼如下:

package main
 
import (
 "fmt"
 "sync"
 "time"
)
 
func main() {
 jobs := make(chan int)
 timeout := make(chan bool)
 var wg sync.WaitGroup
 go func() {
 time.Sleep(time.Second * 3)
 timeout - true
 }()
 go func() {
 for i := 0; ; i++ {
 select {
 case -timeout:
 close(jobs)
 return
 
 default:
 jobs - i
 fmt.Println("produce:", i)
 }
 }
 }()
 wg.Add(1)
 go func() {
 defer wg.Done()
 for i := range jobs {
 fmt.Println("consume:", i)
 }
 }()
 wg.Wait()
}

這樣就可以保證不會往已經關閉的 channel 中發送數據了。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。

您可能感興趣的文章:
  • 基于golang channel實現的輕量級異步任務分發器示例代碼
  • golang中for循環遍歷channel時需要注意的問題詳解
  • golang實現基于channel的通用連接池詳解
  • golang中單向channel的語法介紹
  • golang判斷chan channel是否關閉的方法
  • Golang中channel使用的一些小技巧
  • Golang中channel的原理解讀(推薦)

標簽:阿壩 晉中 東營 滄州 駐馬店 泰安 瀘州 昭通

巨人網絡通訊聲明:本文標題《Golang優雅關閉channel的方法示例》,本文關鍵詞  Golang,優雅,關閉,channel,的,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Golang優雅關閉channel的方法示例》相關的同類信息!
  • 本頁收集關于Golang優雅關閉channel的方法示例的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 双江| 乐清市| 报价| 克什克腾旗| 济源市| 纳雍县| 常宁市| 米易县| 聂荣县| 万荣县| 苏尼特左旗| 吴川市| 新巴尔虎左旗| 大英县| 宜黄县| 桐乡市| 石阡县| 巴塘县| 砀山县| 时尚| 广南县| 嘉义县| 阿克陶县| 黄陵县| 攀枝花市| 土默特左旗| 安乡县| 丁青县| 萨迦县| 屏边| 达日县| 利川市| 铜梁县| 博白县| 景德镇市| 娱乐| 普兰店市| 舟曲县| 桦川县| 义马市| 子长县|