go帶緩沖chan實(shí)現(xiàn)消息隊(duì)列功能
1、Channels 定義
通道是一種支持多類型的管道,您可以通過(guò)它使用通道運(yùn)算符 <- 發(fā)送和接收值。
數(shù)據(jù)沿箭頭方向流動(dòng)。
ch <- v // Send v to channel ch. v := <-ch // Receive from ch, and // assign value to v.
與 maps 和 slices 一樣,通道必須在使用前創(chuàng)建:
ch := make(chan int)
默認(rèn)情況下,發(fā)送和接收阻塞,直到另一方準(zhǔn)備就緒。
這允許 goroutines 在沒有顯式鎖或條件變量的情況下進(jìn)行同步。
package main import "fmt" func sum(s []int, c chan int) { ?? ?sum := 0 ?? ?for _, v := range s { ?? ??? ?sum += v ?? ?} ?? ?c <- sum // send sum to c } func main() { ?? ?s := []int{7, 2, 8, -9, 4, 0} ?? ?c := make(chan int) ?? ?go sum(s[:len(s)/2], c) ?? ?go sum(s[len(s)/2:], c) ?? ?x, y := <-c, <-c // receive from c ?? ?fmt.Println(x, y, x+y) }
2、chan 常用操作
- 無(wú)緩沖區(qū): 存入讀取一次,存入后未取,再存入就會(huì)堵塞,同樣未存,就取也會(huì)堵塞。
- 有緩沖區(qū): 只有當(dāng)緩沖區(qū)滿了,才會(huì)堵塞存;只有緩沖區(qū)空時(shí),才會(huì)堵塞取。
- len(channel) 返回緩沖區(qū)現(xiàn)有數(shù)據(jù)長(zhǎng)度
- cap(channel) 返回緩沖區(qū)的大小
- close(channel) 關(guān)閉 channel,關(guān)閉后,讀取不到數(shù)據(jù)。如下,如果其他協(xié)程關(guān)掉 channel 則會(huì)跳出循環(huán)
3、帶緩沖chan實(shí)現(xiàn)消息隊(duì)列功能
// 監(jiān)測(cè)數(shù)據(jù)結(jié)構(gòu)體 type Msg struct { ?? ?Timestamp?? ?int64 ?? ?Content?? ??? ?string?? ??? ? } // 用 chan 模擬隊(duì)列,隊(duì)列的元素為 Msg 類型 var SyncQueen chan Msg // 必須初始化才能使用。初始化一個(gè)容量為1024的 chan。chan 滿時(shí)會(huì)阻塞 func init() { ?? ?SyncQueen = make(chan Msg, 1024) }
// 隊(duì)列消費(fèi)者 func Consumer() { ?? ?defer func() { ?? ??? ?if err := recover(); err != nil { ?? ??? ??? ?fmt.Println(err) ?? ??? ?} ?? ?}() ?? ?for { ?? ??? ?// chan 內(nèi)無(wú)消息則阻塞 ?? ??? ?msg := <-SyncQueen ?? ??? ?fmt.Println(msg.Content) ?? ?} }
// 隊(duì)列生產(chǎn)者 func Producer() { ?? ?for { ?? ??? ?msg := Msg(time.now().Unix(), "hello") ?? ??? ? ?? ??? ?// 發(fā)送消息到 chan ?? ??? ?SyncQueen <- msg ?? ??? ?time.Sleep(2 time.Second) ?? ?} }
重點(diǎn)
多協(xié)程使用chan是并發(fā)安全的,以下展示一個(gè)簡(jiǎn)單的例子:
// 定義類型為 int 的 chan var chanNums chan int // chan 的消費(fèi)者,用戶后續(xù)多協(xié)程 // 目的:數(shù)組里存儲(chǔ)了10000個(gè)數(shù)字,多個(gè)協(xié)程并行計(jì)算后,把和加起來(lái) func consumer(sum *int) int { ?? ?for { ?? ??? ?v := <-chanNums ?? ??? ?*sum += v ?? ?} } //------------------------------------- func main() { ?? ?var a [10000]int ?? ?for i := 0; i < 10000; i++ { ?? ??? ?a[i] = i + 1 ?? ?} ?? ?chanNums = make(chan int, 10000) ?? ?for i := 0; i < 10000; i++ { ?? ??? ?chanNums <- (i + 1) ?? ?} ?? ?var s1, s2, s3, s4, s5 int = 0, 0, 0, 0, 0 ?? ?go consumer(&s1) ?? ?go consumer(&s2) ?? ?go consumer(&s3) ?? ?go consumer(&s4) ?? ?go consumer(&s5) ?? ?for { ?? ??? ?time.Sleep(5 * time.Second) ?? ??? ?break ?? ?} ?? ?fmt.Println("s1=", s1, "s2=", s2, "s3=", s3, "s4=", s4, "s5=", s5) ?? ?fmt.Println("sum=", s1+s2+s3+s4+s5) } // 輸出 s1= 10818438 s2= 12073966 s3= 9044041 s4= 11509634 s5= 6558921 sum= 50005000
到此這篇關(guān)于go帶緩沖chan實(shí)現(xiàn)消息隊(duì)列功能的文章就介紹到這了,更多相關(guān)go緩沖chan消息隊(duì)列內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
源碼解析gtoken替換jwt實(shí)現(xiàn)sso登錄
這篇文章主要為大家介紹了源碼解析gtoken替換jwt實(shí)現(xiàn)sso登錄的示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06Go實(shí)現(xiàn)MD5加密的三種方法小結(jié)
本文主要介紹了Go實(shí)現(xiàn)MD5加密的三種方法小結(jié),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-03-03Go語(yǔ)言實(shí)現(xiàn)棧與隊(duì)列基本操作學(xué)家
go語(yǔ)言中,并沒有棧與隊(duì)列相關(guān)的數(shù)據(jù)結(jié)構(gòu),但是我們可以借助切片來(lái)實(shí)現(xiàn)棧與隊(duì)列的操作;接下來(lái)我們一起實(shí)現(xiàn)棧與隊(duì)列基本操作,感興趣的可以了解一下2022-11-11go語(yǔ)言開發(fā)中如何優(yōu)雅得關(guān)閉協(xié)程方法
這篇文章主要為大家介紹了go語(yǔ)言開發(fā)中如何優(yōu)雅得關(guān)閉協(xié)程方法詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05go語(yǔ)言日志實(shí)現(xiàn)詳解(打印日志、日志寫入文件和日志切割)
golang內(nèi)置了log包,實(shí)現(xiàn)簡(jiǎn)單的日志服務(wù),下面這篇文章主要給大家介紹了關(guān)于go語(yǔ)言日志實(shí)現(xiàn)(打印日志、日志寫入文件和日志切割)的相關(guān)資料,需要的朋友可以參考下2022-10-10go語(yǔ)言通過(guò)反射創(chuàng)建結(jié)構(gòu)體、賦值、并調(diào)用對(duì)應(yīng)的操作
這篇文章主要介紹了go語(yǔ)言通過(guò)反射創(chuàng)建結(jié)構(gòu)體、賦值、并調(diào)用對(duì)應(yīng)的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05詳解Go語(yǔ)言中Validator庫(kù)的使用方法和用途
github.com/go-playground/validator 是一個(gè) Go 語(yǔ)言的庫(kù),用于對(duì)結(jié)構(gòu)體字段進(jìn)行驗(yàn)證,它提供了一種簡(jiǎn)單而靈活的方式來(lái)定義驗(yàn)證規(guī)則,在這篇文章中,我們將從一個(gè)簡(jiǎn)單的問(wèn)題出發(fā),帶你了解 Validator 庫(kù)的用途,也會(huì)介紹Validator 的基本使用2023-09-09Go?net?http超時(shí)應(yīng)用場(chǎng)景全面詳解
HTTP是一個(gè)復(fù)雜的多階段協(xié)議,因此沒有一個(gè)一刀切的超時(shí)解決方案,在這篇文章中,我將分解您可能需要應(yīng)用超時(shí)的各個(gè)階段,并研究在服務(wù)器端和客戶端上執(zhí)行超時(shí)的不同方法2024-01-01