亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

GoLang?channel關(guān)閉狀態(tài)相關(guān)操作詳解

 更新時間:2022年10月11日 10:50:04   作者:~龐貝  
Channel?和?goroutine?的結(jié)合是?Go?并發(fā)編程的大殺器。而?Channel?的實際應(yīng)用也經(jīng)常讓人眼前一亮,通過與?select,cancel,timer?等結(jié)合,它能實現(xiàn)各種各樣的功能。接下來,我們就要介紹GoLang?channel關(guān)閉狀態(tài)相關(guān)操作

關(guān)于 channel 的使用,有幾點不方便的地方:

1.在不改變 channel 自身狀態(tài)的情況下,無法獲知一個 channel 是否關(guān)閉。

2.關(guān)閉一個 closed channel 會導(dǎo)致 panic。所以,如果關(guān)閉 channel 的一方在不知道 channel 是否處于關(guān)閉狀態(tài)時就去貿(mào)然關(guān)閉 channel 是很危險的事情。

3.向一個 closed channel 發(fā)送數(shù)據(jù)會導(dǎo)致 panic。所以,如果向 channel 發(fā)送數(shù)據(jù)的一方不知道 channel 是否處于關(guān)閉狀態(tài)時就去貿(mào)然向 channel 發(fā)送數(shù)據(jù)是很危險的事情。

一個比較粗糙的檢查 channel 是否關(guān)閉的函數(shù):

func IsClosed(ch <-chan T) bool {
	select {
	case <-ch:
		return true
	default:
	}
	return false
}
func main() {
	c := make(chan T)
	fmt.Println(IsClosed(c)) // false
	close(c)
	fmt.Println(IsClosed(c)) // true
}

看一下代碼,其實存在很多問題。首先,IsClosed 函數(shù)是一個有副作用的函數(shù)。每調(diào)用一次,都會讀出 channel 里的一個元素,改變了 channel 的狀態(tài)。這不是一個好的函數(shù),干活就干活,還順手牽羊!

其次,IsClosed 函數(shù)返回的結(jié)果僅代表調(diào)用那個瞬間,并不能保證調(diào)用之后會不會有其他 goroutine 對它進行了一些操作,改變了它的這種狀態(tài)。例如,IsClosed 函數(shù)返回 true,但這時有另一個 goroutine 關(guān)閉了 channel,而你還拿著這個過時的 “channel 未關(guān)閉”的信息,向其發(fā)送數(shù)據(jù),就會導(dǎo)致 panic 的發(fā)生。當(dāng)然,一個 channel 不會被重復(fù)關(guān)閉兩次,如果 IsClosed 函數(shù)返回的結(jié)果是 true,說明 channel 是真的關(guān)閉了。

有一條廣泛流傳的關(guān)閉 channel 的原則:

don’t close a channel from the receiver side and don’t close a channel if the channel has multiple concurrent senders.

不要從一個 receiver 側(cè)關(guān)閉 channel,也不要在有多個 sender 時,關(guān)閉 channel。

比較好理解,向 channel 發(fā)送元素的就是 sender,因此 sender 可以決定何時不發(fā)送數(shù)據(jù),并且關(guān)閉 channel。但是如果有多個 sender,某個 sender 同樣沒法確定其他 sender 的情況,這時也不能貿(mào)然關(guān)閉 channel。

但是上面所說的并不是最本質(zhì)的,最本質(zhì)的原則就只有一條:

don’t close (or send values to) closed channels.

有兩個不那么優(yōu)雅地關(guān)閉 channel 的方法:

1.使用 defer-recover 機制,放心大膽地關(guān)閉 channel 或者向 channel 發(fā)送數(shù)據(jù)。即使發(fā)生了 panic,有 defer-recover 在兜底。

2.使用 sync.Once 來保證只關(guān)閉一次。

那到底應(yīng)該如何優(yōu)雅地關(guān)閉 channel?

根據(jù) sender 和 receiver 的個數(shù),分下面幾種情況:

  • 一個 sender,一個 receiver
  • 一個 sender, M 個 receiver
  • N 個 sender,一個 reciver
  • N 個 sender, M 個 receiver

對于 1,2,只有一個 sender 的情況就不用說了,直接從 sender 端關(guān)閉就好了,沒有問題。重點關(guān)注第 3,4 種情況。

第 3 種情形下,優(yōu)雅關(guān)閉 channel 的方法是:the only receiver says “please stop sending more” by closing an additional signal channel。

解決方案就是增加一個傳遞關(guān)閉信號的 channel,receiver 通過信號 channel 下達關(guān)閉數(shù)據(jù) channel 指令。senders 監(jiān)聽到關(guān)閉信號后,停止發(fā)送數(shù)據(jù)。代碼如下:

func main() {
	rand.Seed(time.Now().UnixNano())
	const Max = 100000
	const NumSenders = 1000
	dataCh := make(chan int, 100)
	stopCh := make(chan struct{})
	// senders
	for i := 0; i < NumSenders; i++ {
		go func() {
			for {
				select {
				case <- stopCh:
					return
				case dataCh <- rand.Intn(Max):
				}
			}
		}()
	}
	// the receiver
	go func() {
		for value := range dataCh {
			if value == Max-1 {
				fmt.Println("send stop signal to senders.")
				close(stopCh)
				return
			}
			fmt.Println(value)
		}
	}()
	select {
	case <- time.After(time.Hour):
	}
}

這里的 stopCh 就是信號 channel,它本身只有一個 sender,因此可以直接關(guān)閉它。senders 收到了關(guān)閉信號后,select 分支 “case <- stopCh” 被選中,退出函數(shù),不再發(fā)送數(shù)據(jù)。

需要說明的是,上面的代碼并沒有明確關(guān)閉 dataCh。在 Go 語言中,對于一個 channel,如果最終沒有任何 goroutine 引用它,不管 channel 有沒有被關(guān)閉,最終都會被 gc 回收。所以,在這種情形下,所謂的優(yōu)雅地關(guān)閉 channel 就是不關(guān)閉 channel,讓 gc 代勞。

最后一種情況,優(yōu)雅關(guān)閉 channel 的方法是:any one of them says “let’s end the game” by notifying a moderator to close an additional signal channel。

和第 3 種情況不同,這里有 M 個 receiver,如果直接還是采取第 3 種解決方案,由 receiver 直接關(guān)閉 stopCh 的話,就會重復(fù)關(guān)閉一個 channel,導(dǎo)致 panic。因此需要增加一個中間人,M 個 receiver 都向它發(fā)送關(guān)閉 dataCh 的“請求”,中間人收到第一個請求后,就會直接下達關(guān)閉 dataCh 的指令(通過關(guān)閉 stopCh,這時就不會發(fā)生重復(fù)關(guān)閉的情況,因為 stopCh 的發(fā)送方只有中間人一個)。另外,這里的 N 個 sender 也可以向中間人發(fā)送關(guān)閉 dataCh 的請求。

func main() {
	rand.Seed(time.Now().UnixNano())
	const Max = 100000
	const NumReceivers = 10
	const NumSenders = 1000
	dataCh := make(chan int, 100)
	stopCh := make(chan struct{})
	// It must be a buffered channel.
	toStop := make(chan string, 1)
	var stoppedBy string
	// moderator
	go func() {
		stoppedBy = <-toStop
		close(stopCh)
	}()
	// senders
	for i := 0; i < NumSenders; i++ {
		go func(id string) {
			for {
				value := rand.Intn(Max)
				if value == 0 {
					select {
					case toStop <- "sender#" + id:
					default:
					}
					return
				}
				select {
				case <- stopCh:
					return
				case dataCh <- value:
				}
			}
		}(strconv.Itoa(i))
	}
	// receivers
	for i := 0; i < NumReceivers; i++ {
		go func(id string) {
			for {
				select {
				case <- stopCh:
					return
				case value := <-dataCh:
					if value == Max-1 {
						select {
						case toStop <- "receiver#" + id:
						default:
						}
						return
					}

					fmt.Println(value)
				}
			}
		}(strconv.Itoa(i))
	}
	select {
	case <- time.After(time.Hour):
	}
}

代碼里 toStop 就是中間人的角色,使用它來接收 senders 和 receivers 發(fā)送過來的關(guān)閉 dataCh 請求。

這里將 toStop 聲明成了一個 緩沖型的 channel。假設(shè) toStop 聲明的是一個非緩沖型的 channel,那么第一個發(fā)送的關(guān)閉 dataCh 請求可能會丟失。因為無論是 sender 還是 receiver 都是通過 select 語句來發(fā)送請求,如果中間人所在的 goroutine 沒有準備好,那 select 語句就不會選中,直接走 default 選項,什么也不做。這樣,第一個關(guān)閉 dataCh 的請求就會丟失。

如果,我們把 toStop 的容量聲明成 Num(senders) + Num(receivers),那發(fā)送 dataCh 請求的部分可以改成更簡潔的形式:

...
toStop := make(chan string, NumReceivers + NumSenders)
...
			value := rand.Intn(Max)
			if value == 0 {
				toStop <- "sender#" + id
				return
			}
...
				if value == Max-1 {
					toStop <- "receiver#" + id
					return
				}
...

直接向 toStop 發(fā)送請求,因為 toStop 容量足夠大,所以不用擔(dān)心阻塞,自然也就不用 select 語句再加一個 default case 來避免阻塞。

可以看到,這里同樣沒有真正關(guān)閉 dataCh,原樣同第 3 種情況。

到此這篇關(guān)于GoLang channel關(guān)閉狀態(tài)相關(guān)操作詳解的文章就介紹到這了,更多相關(guān)Go channel內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • goland 設(shè)置注釋模板的過程圖文詳解

    goland 設(shè)置注釋模板的過程圖文詳解

    這篇文章主要介紹了goland 設(shè)置注釋模板的過程,本文通過圖文并茂的形式給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友參考下吧
    2020-12-12
  • Golang學(xué)習(xí)筆記(六):struct

    Golang學(xué)習(xí)筆記(六):struct

    這篇文章主要介紹了Golang學(xué)習(xí)筆記(六):struct,本文講解了struct的聲明及初始化、struct的匿名字段(繼承)、method、method繼承和重寫等內(nèi)容,需要的朋友可以參考下
    2015-05-05
  • Golang請求fasthttp實踐

    Golang請求fasthttp實踐

    本文主要介紹了Golang請求fasthttp實踐,文中通過示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-11-11
  • Golang利用casbin實現(xiàn)權(quán)限驗證詳解

    Golang利用casbin實現(xiàn)權(quán)限驗證詳解

    Casbin是一個強大的、高效的開源訪問控制框架,其權(quán)限管理機制支持多種訪問控制模型,Casbin只負責(zé)訪問控制。本文將利用casbin實現(xiàn)權(quán)限驗證功能,需要的可以參考一下
    2023-02-02
  • GO 反射對性能的影響分析

    GO 反射對性能的影響分析

    這篇文章主要為大家介紹了GO 反射對性能的影響分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-01-01
  • 詳解Go語言中獲取文件路徑的不同方法與應(yīng)用場景

    詳解Go語言中獲取文件路徑的不同方法與應(yīng)用場景

    在使用?Go?開發(fā)項目時,估計有不少人遇到過無法正確處理文件路徑的問題,本文將嘗試從簡單到復(fù)雜,詳細介紹?Go?中獲取路徑的不同方法及應(yīng)用場景,希望對大家有所幫助
    2024-02-02
  • Golang使用Gin框架實現(xiàn)HTTP響應(yīng)格式統(tǒng)一處理

    Golang使用Gin框架實現(xiàn)HTTP響應(yīng)格式統(tǒng)一處理

    在gin框架中,我們可以定義一個中間件來處理統(tǒng)一的HTTP響應(yīng)格式,本文主要為大家介紹了具體是怎么定義實現(xiàn)這樣的中間件的,感興趣的小伙伴可以了解一下
    2023-07-07
  • go語言K8S?的?informer機制淺析

    go語言K8S?的?informer機制淺析

    這篇文章為大家主要介紹了go語言K8S?的?informer機制淺析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-10-10
  • Go 實現(xiàn)HTTP中間人代理的操作

    Go 實現(xiàn)HTTP中間人代理的操作

    這篇文章主要介紹了Go 實現(xiàn)HTTP中間人代理的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • Golang 類型轉(zhuǎn)換的實現(xiàn)(斷言、強制、顯式類型)

    Golang 類型轉(zhuǎn)換的實現(xiàn)(斷言、強制、顯式類型)

    將一個值從一種類型轉(zhuǎn)換到另一種類型,便發(fā)生了類型轉(zhuǎn)換,在go可以分為斷言、強制、顯式類型轉(zhuǎn)換,本文就詳細的介紹一下這就幾種轉(zhuǎn)換方式,具有一定的參考價值,感興趣的可以了解一下
    2023-09-09

最新評論