0

0

Go语言实现简单消息队列_Go异步处理项目示例

P粉602998670

P粉602998670

发布时间:2026-01-15 10:38:03

|

761人浏览过

|

来源于php中文网

原创

使用无缓冲 channel 实现队列时,若生产者与消费者未协同启动(如仅生产无消费,或消费先阻塞等待),所有 goroutine 会因 channel 操作阻塞而休眠,触发“all goroutines are asleep - deadlock”错误。

go语言实现简单消息队列_go异步处理项目示例

chan 实现队列时,为什么一跑就报 fatal error: all goroutines are asleep - deadlock

这是最常踩的坑:裸用无缓冲 chan 且没配好生产者/消费者节奏。比如只启了生产者往 chan string 写,但没启消费者读,或消费者先启动却等不到数据,goroutine 全卡住,Go 运行时直接 panic。

  • 必须确保至少一个 goroutine 在 range 等待读,另一个在 写(或用 select 防死等)
  • 别用无缓冲 channel 做“队列”——它本质是同步点,不是缓存;要用就选带缓冲的:messages := make(chan string, 10)
  • 如果要支持“空队列时等待”,不能只靠 ,得用 select + time.After 或配合 sync.Cond 手动管理状态

MessageQueue 结构体封装里,sync.Mutex 不是用来保护 chan

Go 的 channel 本身是并发安全的,sync.Mutex 在这里不是必需品。加它的真正目的是为后续扩展留钩子:比如加消息计数器、记录入队时间、统计积压量,或者将来把内存队列换成 Redis 后端时,统一加日志或重试逻辑。

  • 当前仅用 channel 时,EnqueueDequeue 方法内部无需加锁
  • 但如果结构体里加了 len 字段来实时统计长度,那每次读写就得用 mu.Lock() 保护
  • 错误做法:给 channel 加锁后还用 len(ch) 判断是否满——这不可靠,因为 len 是瞬时快照,且对 buffered channel 才有意义

select 处理超时和非阻塞,比裸写 更健壮

真实场景中,你不能让生产者无限期卡在 messages 上,尤其当消费者处理慢、队列满时。用 select 可明确控制行为边界。

  • 非阻塞入队(丢弃策略):
    select {
    case messages <- "新消息":
        fmt.Println("入队成功")
    default:
        fmt.Println("队列已满,丢弃")
    }
  • 带超时入队(降级策略):
    select {
    case messages <- "新消息":
        fmt.Println("入队成功")
    case <-time.After(500 * time.Millisecond):
        fmt.Println("超时,放弃发送")
    }
  • 消费者侧同理:用 select 包裹 ,避免单个 goroutine 挂死导致整条消费链停滞

什么时候该从内存 chan 切到 Redis 或 Kafka

纯内存队列适合开发验证、单机轻量任务(如日志聚合、内部通知)。一旦出现以下任一情况,就得换:

Reecho睿声
Reecho睿声

Reecho AI:超拟真语音合成与瞬时语音克隆平台

下载

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

  • 需要消息持久化(进程重启不丢数据)→ 必须上 Redis(LPUSH/BRPOP)或 Kafka
  • 消费者数量动态伸缩,或跨机器部署 → 内存 chan 无法共享,Redis 或 Kafka 提供中心化分发
  • 要求精确一次(exactly-once)语义、死信队列、延时消息 → 内存方案几乎没法可靠实现
  • 流量突发远超内存 buffer 容量(比如秒杀场景)→ Redis 的内存+磁盘混合模式或 Kafka 的分区机制更稳

别等到线上崩了才换——只要服务要长期运行、有外部依赖、或用户量破千,就该默认选 Redis 方案,哪怕只是本地 localhost:6379

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

187

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

271

2023.10.25

Go中Type关键字的用法
Go中Type关键字的用法

Go中Type关键字的用法有定义新的类型别名或者创建新的结构体类型。本专题为大家提供Go相关的文章、下载、课程内容,供大家免费下载体验。

233

2023.09.06

go怎么实现链表
go怎么实现链表

go通过定义一个节点结构体、定义一个链表结构体、定义一些方法来操作链表、实现一个方法来删除链表中的一个节点和实现一个方法来打印链表中的所有节点的方法实现链表。

444

2023.09.25

go语言编程软件有哪些
go语言编程软件有哪些

go语言编程软件有Go编译器、Go开发环境、Go包管理器、Go测试框架、Go文档生成器、Go代码质量工具和Go性能分析工具等。本专题为大家提供go语言相关的文章、下载、课程内容,供大家免费下载体验。

246

2023.10.13

0基础如何学go语言
0基础如何学go语言

0基础学习Go语言需要分阶段进行,从基础知识到实践项目,逐步深入。php中文网给大家带来了go语言相关的教程以及文章,欢迎大家前来学习。

693

2023.10.26

Go语言实现运算符重载有哪些方法
Go语言实现运算符重载有哪些方法

Go语言不支持运算符重载,但可以通过一些方法来模拟运算符重载的效果。使用函数重载来模拟运算符重载,可以为不同的类型定义不同的函数,以实现类似运算符重载的效果,通过函数重载,可以为不同的类型实现不同的操作。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

191

2024.02.23

Java 桌面应用开发(JavaFX 实战)
Java 桌面应用开发(JavaFX 实战)

本专题系统讲解 Java 在桌面应用开发领域的实战应用,重点围绕 JavaFX 框架,涵盖界面布局、控件使用、事件处理、FXML、样式美化(CSS)、多线程与UI响应优化,以及桌面应用的打包与发布。通过完整示例项目,帮助学习者掌握 使用 Java 构建现代化、跨平台桌面应用程序的核心能力。

63

2026.01.14

热门下载

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

精品课程

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

共32课时 | 3.7万人学习

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号