溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

怎么使用go帶緩沖chan實現(xiàn)消息隊列功能

發(fā)布時間:2023-02-28 14:28:55 來源:億速云 閱讀:206 作者:iii 欄目:開發(fā)技術

本篇內(nèi)容介紹了“怎么使用go帶緩沖chan實現(xiàn)消息隊列功能”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!

1、Channels 定義

通道是一種支持多類型的管道,您可以通過它使用通道運算符 <- 發(fā)送和接收值。

數(shù)據(jù)沿箭頭方向流動。

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)

默認情況下,發(fā)送和接收阻塞,直到另一方準備就緒。
這允許 goroutines 在沒有顯式鎖或條件變量的情況下進行同步。

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 常用操作

  • 無緩沖區(qū): 存入讀取一次,存入后未取,再存入就會堵塞,同樣未存,就取也會堵塞。

  • 有緩沖區(qū): 只有當緩沖區(qū)滿了,才會堵塞存;只有緩沖區(qū)空時,才會堵塞取。

  • len(channel) 返回緩沖區(qū)現(xiàn)有數(shù)據(jù)長度

  • cap(channel) 返回緩沖區(qū)的大小

  • close(channel) 關閉 channel,關閉后,讀取不到數(shù)據(jù)。如下,如果其他協(xié)程關掉 channel 則會跳出循環(huán)

3、帶緩沖chan實現(xiàn)消息隊列功能

// 監(jiān)測數(shù)據(jù)結構體
type Msg struct {
    Timestamp    int64
    Content        string        
}

// 用 chan 模擬隊列,隊列的元素為 Msg 類型
var SyncQueen chan Msg

// 必須初始化才能使用。初始化一個容量為1024的 chan。chan 滿時會阻塞
func init() {
    SyncQueen = make(chan Msg, 1024)
}
// 隊列消費者
func Consumer() {
    defer func() {
        if err := recover(); err != nil {
            fmt.Println(err)
        }
    }()

    for {
        // chan 內(nèi)無消息則阻塞
        msg := <-SyncQueen

        fmt.Println(msg.Content)
    }
}
// 隊列生產(chǎn)者
func Producer() {
    for {
        msg := Msg(time.now().Unix(), "hello")
        
        // 發(fā)送消息到 chan
        SyncQueen <- msg
        time.Sleep(2 time.Second)
    }
}

重點

多協(xié)程使用chan是并發(fā)安全的,以下展示一個簡單的例子:

// 定義類型為 int 的 chan
var chanNums chan int

// chan 的消費者,用戶后續(xù)多協(xié)程
// 目的:數(shù)組里存儲了10000個數(shù)字,多個協(xié)程并行計算后,把和加起來
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

“怎么使用go帶緩沖chan實現(xiàn)消息隊列功能”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關的知識可以關注億速云網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI