go带缓冲chan实现消息队列功能

 更新时间:2023年02月28日 10:28:03   作者:wecode.fun  
本文主要介绍了go带缓冲chan实现消息队列功能,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

1、Channels 定义

通道是一种支持多类型的管道,您可以通过它使用通道运算符 <- 发送和接收值。

数据沿箭头方向流动。

ch <- v    // Send v to channel ch.
v := <-ch  // Receive from ch, and
           // assign value to v.

与 maps 和 slices 一样,通道必须在使用前创建:

ch := make(chan int)

默认情况下,发送和接收阻塞,直到另一方准备就绪。
这允许 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 常用操作

  • 无缓冲区: 存入读取一次,存入后未取,再存入就会堵塞,同样未存,就取也会堵塞。
  • 有缓冲区: 只有当缓冲区满了,才会堵塞存;只有缓冲区空时,才会堵塞取。
  • len(channel) 返回缓冲区现有数据长度
  • cap(channel) 返回缓冲区的大小
  • close(channel) 关闭 channel,关闭后,读取不到数据。如下,如果其他协程关掉 channel 则会跳出循环

3、带缓冲chan实现消息队列功能

// 监测数据结构体
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 内无消息则阻塞
        msg := <-SyncQueen

        fmt.Println(msg.Content)
    }
}
// 队列生产者
func Producer() {
    for {
        msg := Msg(time.now().Unix(), "hello")
        
        // 发送消息到 chan
        SyncQueen <- msg
        time.Sleep(2 time.Second)
    }
}

重点

多协程使用chan是并发安全的,以下展示一个简单的例子:

// 定义类型为 int 的 chan
var chanNums chan int

// chan 的消费者,用户后续多协程
// 目的:数组里存储了10000个数字,多个协程并行计算后,把和加起来
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实现消息队列功能的文章就介绍到这了,更多相关go缓冲chan消息队列内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Go语言sync.Cond基本使用及原理示例详解

    Go语言sync.Cond基本使用及原理示例详解

    这篇文章主要为大家介绍了Go语言sync.Cond基本使用及原理示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • 源码解析gtoken替换jwt实现sso登录

    源码解析gtoken替换jwt实现sso登录

    这篇文章主要为大家介绍了源码解析gtoken替换jwt实现sso登录的示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-06-06
  • Go操作Kafka和Etcd方法详解

    Go操作Kafka和Etcd方法详解

    这篇文章主要为大家介绍了Go操作Kafka和Etcd方法详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-09-09
  • Go实现MD5加密的三种方法小结

    Go实现MD5加密的三种方法小结

    本文主要介绍了Go实现MD5加密的三种方法小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-03-03
  • Go语言实现栈与队列基本操作学家

    Go语言实现栈与队列基本操作学家

    go语言中,并没有栈与队列相关的数据结构,但是我们可以借助切片来实现栈与队列的操作;接下来我们一起实现栈与队列基本操作,感兴趣的可以了解一下
    2022-11-11
  • go语言开发中如何优雅得关闭协程方法

    go语言开发中如何优雅得关闭协程方法

    这篇文章主要为大家介绍了go语言开发中如何优雅得关闭协程方法详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-05-05
  • go语言日志实现详解(打印日志、日志写入文件和日志切割)

    go语言日志实现详解(打印日志、日志写入文件和日志切割)

    golang内置了log包,实现简单的日志服务,下面这篇文章主要给大家介绍了关于go语言日志实现(打印日志、日志写入文件和日志切割)的相关资料,需要的朋友可以参考下
    2022-10-10
  • go语言通过反射创建结构体、赋值、并调用对应的操作

    go语言通过反射创建结构体、赋值、并调用对应的操作

    这篇文章主要介绍了go语言通过反射创建结构体、赋值、并调用对应的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2021-05-05
  • 详解Go语言中Validator库的使用方法和用途

    详解Go语言中Validator库的使用方法和用途

    github.com/go-playground/validator 是一个 Go 语言的库,用于对结构体字段进行验证,它提供了一种简单而灵活的方式来定义验证规则,在这篇文章中,我们将从一个简单的问题出发,带你了解 Validator 库的用途,也会介绍Validator 的基本使用
    2023-09-09
  • Go net http超时应用场景全面详解

    Go net http超时应用场景全面详解

    HTTP是一个复杂的多阶段协议,因此没有一个一刀切的超时解决方案,在这篇文章中,我将分解您可能需要应用超时的各个阶段,并研究在服务器端和客户端上执行超时的不同方法
    2024-01-01

最新评论