欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Golang優(yōu)雅關(guān)閉channel的方法示例

 更新時間:2017年11月19日 09:52:39   作者:小談  
Goroutine和channel是Go在“并發(fā)”方面兩個核心feature,下面這篇文章主要給大家介紹了關(guān)于Golang如何優(yōu)雅關(guān)閉channel的相關(guān)資料,文中通過示例代碼介紹的非常詳細,需要的朋友可以參考解決,下面來一起看看吧。

前言

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

關(guān)于Go channel設(shè)計和規(guī)范的批評:

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

所以Golang 內(nèi)建的 close 方法可以關(guān)閉 channel,如果往已經(jīng)關(guān)閉的 channel 發(fā)送數(shù)據(jù),則會報錯:panic: close of closed channel.

看如下代碼,在一段時間內(nèi),生產(chǎn)者可以不斷往 channel 寫入數(shù)據(jù),消費者進行處理,一段時間后 channel 關(guān)閉了,這個時候如果還有數(shù)據(jù)往 channel 發(fā)送,程序就會報錯。

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

如何優(yōu)雅關(guān)閉 channel

那么在往通道發(fā)數(shù)據(jù)前如何判斷通道是否關(guān)閉呢?

1._,ok := <- jobs

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

2.使用 select 方式

再創(chuàng)建一個 channel,叫做 timeout,如果超時往這個 channel 發(fā)送 true,在生產(chǎn)者發(fā)送數(shù)據(jù)給 jobs 的 channel,用 select 監(jiān)聽 timeout,如果超時則關(guān)閉 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()
}

這樣就可以保證不會往已經(jīng)關(guān)閉的 channel 中發(fā)送數(shù)據(jù)了。

總結(jié)

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

相關(guān)文章

  • Go語言同步等待組sync.WaitGroup結(jié)構(gòu)體對象方法詳解

    Go語言同步等待組sync.WaitGroup結(jié)構(gòu)體對象方法詳解

    這篇文章主要為大家介紹了Go語言同步等待組sync.WaitGroup結(jié)構(gòu)體對象方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • 詳解golang中模板的常用語法

    詳解golang中模板的常用語法

    這篇文章主要介紹了golang模板中的常用語法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-08-08
  • Golang字符串的拼接方法匯總

    Golang字符串的拼接方法匯總

    字符串拼接在日常開發(fā)中是很常見的需求,今天我們來探討下如何用golang來實現(xiàn)字符串的拼接
    2018-10-10
  • Go并發(fā)編程之死鎖與活鎖的案例分析

    Go并發(fā)編程之死鎖與活鎖的案例分析

    死鎖就是在并發(fā)程序中,兩個或多個線程彼此等待對方完成操作,從而導(dǎo)致它們都被阻塞,并無限期地等待對方完成;活鎖就是程序一直在運行,但是無法取得進展。本文將從一些案例出發(fā),分析一下它們,希望對大家有所幫助
    2023-04-04
  • gin自定義中間件解決requestBody不可重讀(請求體取值)

    gin自定義中間件解決requestBody不可重讀(請求體取值)

    這篇文章主要介紹了gin自定義中間件解決requestBody不可重讀,確保控制器能夠獲取請求體值,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-10-10
  • Golang 刪除文件并遞歸刪除空目錄的操作

    Golang 刪除文件并遞歸刪除空目錄的操作

    這篇文章主要介紹了Golang 刪除文件并遞歸刪除空目錄的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • golang中validator包的使用教程

    golang中validator包的使用教程

    Validator 實際上是一個驗證工具,屬于 golang 的第三方包,這個包中使用了各種反射技巧來提供了各種校驗和約束數(shù)據(jù)的方式方法,下面就跟隨小編一起來學習一下validator包的使用吧
    2023-09-09
  • Go語言實現(xiàn)二進制與十進制互轉(zhuǎn)的示例代碼

    Go語言實現(xiàn)二進制與十進制互轉(zhuǎn)的示例代碼

    這篇文章主要和大家詳細介紹了Go語言中實現(xiàn)二進制與十進制互相轉(zhuǎn)換的示例代碼,文中的代碼簡潔易懂,感興趣的小伙伴可以跟隨小編一起學習一下
    2023-05-05
  • Go語言學習之JSON編碼解析與使用

    Go語言學習之JSON編碼解析與使用

    這篇文章主要為大家詳細介紹了Go語言中JSON編碼的解析與使用已經(jīng)JSON與Map、結(jié)構(gòu)體的互相轉(zhuǎn)化,文中的示例代碼講解詳細,需要的可以參考一下
    2023-02-02
  • Go接口的用法詳解

    Go接口的用法詳解

    本文主要介紹了Go接口的用法詳解,包括定義接口、實現(xiàn)接口、使用接口、空接口等,通過接口,可以實現(xiàn)多態(tài)性,即一個對象可以實現(xiàn)多個接口,從而實現(xiàn)不同接口的行為,感興趣的可以了解一下
    2023-11-11

最新評論