Go語言帶緩沖的通道的使用
Go語言中有緩沖的通道(buffered channel)是一種在被接收前能存儲一個(gè)或者多個(gè)值的通道。這種類型的通道并不強(qiáng)制要求 goroutine 之間必須同時(shí)完成發(fā)送和接收。通道會阻塞發(fā)送和接收動作的條件也會不同。只有在通道中沒有要接收的值時(shí),接收動作才會阻塞。只有在通道沒有可用緩沖區(qū)容納被發(fā)送的值時(shí),發(fā)送動作才會阻塞。
這導(dǎo)致有緩沖的通道和無緩沖的通道之間的一個(gè)很大的不同:無緩沖的通道保證進(jìn)行發(fā)送和接收的 goroutine 會在同一時(shí)間進(jìn)行數(shù)據(jù)交換;有緩沖的通道沒有這種保證。
在無緩沖通道的基礎(chǔ)上,為通道增加一個(gè)有限大小的存儲空間形成帶緩沖通道。帶緩沖通道在發(fā)送時(shí)無需等待接收方接收即可完成發(fā)送過程,并且不會發(fā)生阻塞,只有當(dāng)存儲空間滿時(shí)才會發(fā)生阻塞。同理,如果緩沖通道中有數(shù)據(jù),接收時(shí)將不會發(fā)生阻塞,直到通道中沒有數(shù)據(jù)可讀時(shí),通道將會再度阻塞。
無緩沖通道保證收發(fā)過程同步。無緩沖收發(fā)過程類似于快遞員給你電話讓你下樓取快遞,整個(gè)遞交快遞的過程是同步發(fā)生的,你和快遞員不見不散。但這樣做快遞員就必須等待所有人下樓完成操作后才能完成所有投遞工作。如果快遞員將快遞放入快遞柜中,并通知用戶來取,快遞員和用戶就成了異步收發(fā)過程,效率可以有明顯的提升。帶緩沖的通道就是這樣的一個(gè)“快遞柜”。
創(chuàng)建帶緩沖通道
如何創(chuàng)建帶緩沖的通道呢?參見如下代碼:
通道實(shí)例 := make(chan 通道類型, 緩沖大小)
- 通道類型:和無緩沖通道用法一致,影響通道發(fā)送和接收的數(shù)據(jù)類型。
- 緩沖大?。簺Q定通道最多可以保存的元素?cái)?shù)量。
- 通道實(shí)例:被創(chuàng)建出的通道實(shí)例。
下面通過一個(gè)例子中來理解帶緩沖通道的用法,參見下面的代碼:
package main import "fmt" func main() { // 創(chuàng)建一個(gè)3個(gè)元素緩沖大小的整型通道 ch := make(chan int, 3) // 查看當(dāng)前通道的大小 fmt.Println(len(ch)) // 發(fā)送3個(gè)整型元素到通道 ch <- 1 ch <- 2 ch <- 3 // 查看當(dāng)前通道的大小 fmt.Println(len(ch)) }
代碼輸出如下:
0
3
代碼說明如下:
- 第 8 行,創(chuàng)建一個(gè)帶有 3 個(gè)元素緩沖大小的整型類型的通道。
- 第 11 行,查看當(dāng)前通道的大小。帶緩沖的通道在創(chuàng)建完成時(shí),內(nèi)部的元素是空的,因此使用 len() 獲取到的返回值為 0。
- 第 14~16 行,發(fā)送 3 個(gè)整型元素到通道。因?yàn)槭褂昧司彌_通道。即便沒有 goroutine 接收,發(fā)送者也不會發(fā)生阻塞。
- 第 19 行,由于填充了 3 個(gè)通道,此時(shí)的通道長度變?yōu)?3。
阻塞條件
帶緩沖通道在很多特性上和無緩沖通道是類似的。無緩沖通道可以看作是長度永遠(yuǎn)為 0 的帶緩沖通道。因此根據(jù)這個(gè)特性,帶緩沖通道在下面列舉的情況下依然會發(fā)生阻塞:
- 帶緩沖通道被填滿時(shí),嘗試再次發(fā)送數(shù)據(jù)時(shí)發(fā)生阻塞。
- 帶緩沖通道為空時(shí),嘗試接收數(shù)據(jù)時(shí)發(fā)生阻塞。
為什么Go語言對通道要限制長度而不提供無限長度的通道?
我們知道通道(channel)是在兩個(gè) goroutine 間通信的橋梁。使用 goroutine 的代碼必然有一方提供數(shù)據(jù),一方消費(fèi)數(shù)據(jù)。當(dāng)提供數(shù)據(jù)一方的數(shù)據(jù)供給速度大于消費(fèi)方的數(shù)據(jù)處理速度時(shí),如果通道不限制長度,那么內(nèi)存將不斷膨脹直到應(yīng)用崩潰。因此,限制通道的長度有利于約束數(shù)據(jù)提供方的供給速度,供給數(shù)據(jù)量必須在消費(fèi)方處理量+通道長度的范圍內(nèi),才能正常地處理數(shù)據(jù)。
到此這篇關(guān)于Go語言帶緩沖的通道的使用的文章就介紹到這了,更多相關(guān)Go語言帶緩沖通道內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- golang關(guān)閉chan通道的方法示例
- Go語言無緩沖的通道的使用
- 一文教你Golang如何正確關(guān)閉通道
- 詳解Golang中的通道機(jī)制與應(yīng)用
- Go語言?Channel通道詳解
- golang使用通道時(shí)需要注意的一些問題
- Go語言通道之無緩沖通道與緩沖通道詳解
- 一文帶你掌握Golang基礎(chǔ)之通道
- 超實(shí)用的Golang通道指南之輕松實(shí)現(xiàn)并發(fā)編程
- Go語言單向通道的實(shí)現(xiàn)
- 淺談golang通道類型
- Golang通道的無阻塞讀寫的方法示例
- Golang通道阻塞情況與通道無阻塞實(shí)現(xiàn)小結(jié)
相關(guān)文章
詳解golang consul-grpc 服務(wù)注冊與發(fā)現(xiàn)
這篇文章主要介紹了詳解golang consul-grpc 服務(wù)注冊與發(fā)現(xiàn),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-06-064個(gè)場景教會你Go中Goroutine和通道是怎么用的
本篇給出了4個(gè)在運(yùn)維開發(fā)工作中較為常見的且也是比較典型的場景,通過這些場景來帶大家掌握Go中Goroutine和通道是怎么使用的,需要的可以學(xué)習(xí)一下2023-05-05Go語言中Struct與繼承與匿名字段和內(nèi)嵌結(jié)構(gòu)體全面詳解
這篇文章主要介紹了Go語言中Struct與繼承與匿名字段和內(nèi)嵌結(jié)構(gòu)體,Go語言中通過結(jié)構(gòu)體的內(nèi)嵌再配合接口比面向?qū)ο缶哂懈叩臄U(kuò)展性和靈活性,感興趣的可以了解一下2023-04-04詳解Go如何實(shí)現(xiàn)協(xié)程并發(fā)執(zhí)行
線程是通過本地隊(duì)列,全局隊(duì)列或者偷其它線程的方式來獲取協(xié)程的,目前看來,線程運(yùn)行完一個(gè)協(xié)程后再從隊(duì)列中獲取下一個(gè)協(xié)程執(zhí)行,還只是順序執(zhí)行協(xié)程的,而多個(gè)線程一起這么運(yùn)行也能達(dá)到并發(fā)的效果,接下來就給給大家詳細(xì)介紹一下Go如何實(shí)現(xiàn)協(xié)程并發(fā)執(zhí)行2023-08-08golang獲取prometheus數(shù)據(jù)(prometheus/client_golang包)
本文主要介紹了使用Go語言的prometheus/client_golang包來獲取Prometheus監(jiān)控?cái)?shù)據(jù),具有一定的參考價(jià)值,感興趣的可以了解一下2025-03-03