0

0

如何在Golang中实现简单消息队列_Golang channel与goroutine应用实践

P粉602998670

P粉602998670

发布时间:2026-02-03 12:50:02

|

532人浏览过

|

来源于php中文网

原创

用 make(chan T, N) 创建带缓冲 channel 就是最简消息队列:天然并发安全、底层环形队列+goroutine等待队列,sendx/recvx自动管理读写,qcount实时记录长度;N需按场景选取,日志100~1000、任务10~50、通知1~5;无缓冲channel非队列,是同步点。

如何在golang中实现简单消息队列_golang channel与goroutine应用实践

make(chan T, N) 创建带缓冲 channel 就是最简消息队列

不需要额外依赖、不写锁、不造轮子——Go 原生 channel 加上缓冲区,就是最轻量、最安全的内存级消息队列。它天然并发安全,底层是环形队列 + goroutine 等待队列,sendxrecvx 指针自动管理读写位置,qcount 实时记录长度,比手写锁保护的 slice 队列更可靠。

  • 缓冲大小 N 不是“越多越好”:设太大(如 10000)可能掩盖背压问题,导致 OOM;太小(如 1)又容易让生产者频繁阻塞
  • 典型取值参考:日志采集可设 100~1000,任务调度建议 10~50,纯通知类消息 1~5 即可
  • 切忌用无缓冲 make(chan T) 当队列——它本质是同步点,不是队列,一发一收才继续,无法缓存

必须显式 close() 并用 val, ok := 判断关闭状态

很多初学者在消费者侧直接写 for msg := range ch,看似简洁,但一旦生产者 panic 或提前退出没关 channel,消费者就永久阻塞在 range 上,整个 goroutine 泄漏。真实场景中,生产者生命周期常不可控(比如 HTTP handler 中启动),必须主动管理关闭时机。

  • 关闭前确保所有发送已完成,否则会 panic:panic: send on closed channel
  • 接收端永远用 msg, ok := ,ok == false 表示 channel 已关闭且无剩余数据
  • 不要在多个 goroutine 中重复调用 close(ch),会 panic:panic: close of closed channel
go func() {
    defer close(taskQueue) // 在 defer 中 close 更安全
    for i := 0; i < 10; i++ {
        taskQueue <- Task{ID: i}
        time.Sleep(100 * time.Millisecond)
    }
}()

for { task, ok := <-taskQueue if !ok { break // 退出循环 } process(task) }

select + default 实现非阻塞发送,避免生产者卡死

当队列满时,ch 会阻塞,如果生产者是关键路径(比如 HTTP 请求处理),卡住等于服务不可用。这时候不能靠增大缓冲区硬扛,而要用 select 做快速失败或降级。

  • select 是 Go 的多路复用原语,配合 default 可实现“尝试发送,失败就走其他逻辑”
  • 常见降级策略:打日志告警、丢弃低优先级消息、写入本地磁盘暂存、返回客户端限流响应
  • 注意:不能在 default 分支里加 time.Sleep 循环重试,那会退化成忙等,CPU 拉满
select {
case taskQueue <- task:
    // 发送成功
default:
    log.Warn("task queue full, dropping task")
    // 或:go diskBuffer.Write(task) —— 异步落盘
}

并发消费时,别用 range 启动无限 worker,要配 sync.WaitGroup

网上常见错误模式:for i := 0; i 。这看似启动了 4 个消费者,但实际所有 goroutine 共享同一个 ch 变量,且没有机制等待它们结束——主 goroutine 一退出,整个程序就终止,正在处理的任务直接被杀掉。

DeepBrain
DeepBrain

AI视频生成工具,ChatGPT +生成式视频AI =你可以制作伟大的视频!

下载

立即学习go语言免费学习笔记(深入)”;

  • 必须用 sync.WaitGroup 记录活跃 worker 数,并在每个 worker 结束时 wg.Done()
  • 关闭 channel 后,worker 自然退出 for range,此时再 wg.Wait() 才算真正完成
  • 若需支持动态增减 worker,就得换用更复杂的信号量或上下文控制,基础场景不推荐
var wg sync.WaitGroup
for i := 0; i < 4; i++ {
    wg.Add(1)
    go func(workerID int) {
        defer wg.Done()
        for task := range taskQueue {
            processTask(task, workerID)
        }
    }(i)
}

// 生产完成后关闭 close(taskQueue) wg.Wait() // 等所有 worker 处理完才继续

真正难的不是写出来,而是想清楚:这条消息丢了能不能接受?延迟 200ms 是否在 SLA 内?队列满时该拒绝还是排队?这些决策比语法更重要——channel 是工具,业务语义才是核心。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

184

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

230

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

344

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

210

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

397

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

282

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

194

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

540

2025.06.17

c语言中/相关合集
c语言中/相关合集

本专题整合了c语言中/的用法、含义解释。阅读专题下面的文章了解更多详细内容。

0

2026.02.03

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Go 教程
Go 教程

共32课时 | 4.5万人学习

Go语言实战之 GraphQL
Go语言实战之 GraphQL

共10课时 | 0.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号