RocketMQ 5.x + Spring Boot 发送消息失败全解析
从 gRPC 端口到 Topic 配置的完整踩坑实录
关键词 :RocketMQ 5.x、Spring Boot、gRPC、Producer FAILED、No topic route info
适用版本:RocketMQ 5.3.x + rocketmq-client-java 5.x
一、问题背景
在使用 RocketMQ 5.3.3 搭配 Spring Boot 开发消息生产者时,服务启动正常、控制台可访问、Topic 也"看似"已经创建,但在真正发送消息时却频繁失败,典型报错如下:
text
Expected the service ProducerImpl-0 [FAILED] to be RUNNING
或:
text
No topic route info in name server for the topic: rewre
更让人困惑的是:
- RocketMQ Console 可以正常访问
- NameServer 和 Broker 都已启动
- Topic 在控制台里"存在"
- Producer 代码没有明显错误
但消息就是发不出去。
本文将完整复盘这个问题,并给出 RocketMQ 5.x 下的正确使用姿势。
二、RocketMQ 5.x 与 4.x 的本质差异(很多坑从这里开始)
1️⃣ 通信协议发生了根本变化
| 版本 | 通信方式 |
|---|---|
| RocketMQ 4.x | 自定义 TCP 协议 |
| RocketMQ 5.x | gRPC |
👉 RocketMQ 5.x 的 Java Client 完全基于 gRPC
这意味着:
- NameServer
- Broker
- Client(Producer / Consumer)
三者之间必须能通过 gRPC 端口通信
2️⃣ RocketMQ 5.x 新增的关键端口
以 RocketMQ 5.3.3 为例:
NameServer
| 端口 | 作用 |
|---|---|
| 9876 | 兼容旧协议(Console / Admin) |
| 8081 | ✅ gRPC 端口(5.x Client 必须) |
Broker
| 端口 | 作用 |
|---|---|
| 10911 | 旧协议 |
| 8080 | ✅ gRPC 端口(Producer / Consumer 真正使用) |
⚠️ 如果只暴露 9876 / 10911,而没暴露 8081 / 8080:
- Console 能用
- Topic 能看
- Producer 一发消息就失败
三、问题一:Producer FAILED ------ 实际是 gRPC 不通
典型异常
text
org.apache.rocketmq.shaded.io.grpc.StatusRuntimeException:
UNAVAILABLE: Network closed for unknown reason
本质原因
RocketMQ 5.x Client 通过 gRPC 连接 Broker,但 Broker 的 gRPC 端口未暴露或不可达
解决方式
在 Docker / 容器环境中,必须显式暴露 gRPC 端口:
yaml
# NameServer
ports:
- "9876:9876"
- "8081:8081"
# Broker
ports:
- "10911:10911"
- "8080:8080"
四、问题二:No topic route info ------ 真正的大坑
在 gRPC 问题解决后,紧接着会遇到第二个错误:
text
No topic route info in name server for the topic: rewre
表面含义
NameServer 中没有该 Topic 的路由信息
但问题并不只是"没建 topic"这么简单。
五、致命配置错误:Producer 默认 Topic + 代码显式 Topic 混用
1️⃣ 错误示例(问题根源)
yaml
rocketmq:
producer:
endpoints: 127.0.0.1:8081
group: dromara-producer-group
topic: rewre # ❌ Producer 级别默认 topic
yaml
demo:
rocketmq:
normal-topic: normal-topic
java
rocketMQClientTemplate.syncSendNormalMessage(normalTopic, message);
日志打印的是:
text
发送普通消息到主题: normal-topic
但异常却是:
text
No topic route info for topic: rewre
2️⃣ 为什么会这样?
这是 RocketMQ 5.x Client 的一个"隐式行为":
如果在
rocketmq.producer.topic中配置了 topic,
Producer 初始化和首次发送时,会优先拉取该 topic 的路由信息
即使你在 send() 时传入了别的 topic:
java
syncSendNormalMessage("normal-topic", ...)
👉 Client 仍然会先去 NameServer 查询 rewre 的路由
如果 rewre 不存在:
40402 No topic route info
六、正确姿势:两种方案,只能选一种
✅ 方案一(强烈推荐):删除 Producer 默认 topic
正确配置
yaml
rocketmq:
producer:
endpoints: 127.0.0.1:8081
group: dromara-producer-group
timeout: 10000
max-retry-times: 2
enable-ssl: false
Topic 全部由代码控制
java
syncSendNormalMessage(normalTopic, message);
syncSendDelayMessage(delayTopic, message);
syncSendFifoMessage(fifoTopic, message);
手动创建 topic(必须)
bash
mqadmin updatetopic -n rmqnamesrv:9876 -c DefaultCluster -t normal-topic
👉 这是官方示例和生产环境最推荐的方式
⚠️ 方案二(不推荐):只使用一个固定 topic
如果配置了:
yaml
rocketmq:
producer:
topic: rewre
那么你必须:
- 代码中 只能发
rewre - NameServer 中 必须存在
rewre
java
syncSendNormalMessage("rewre", message);
❌ 不适合多 topic、延时、事务、顺序消息场景
七、RocketMQ 5.x 的几个重要认知纠正
1️⃣ RocketMQ 5.x 不会自动创建 Topic
和 4.x 不同:
Topic 必须提前创建,否则必定 40402
2️⃣ Console 看到 ≠ Client 能用
- Console / mqadmin:走旧协议
- Producer / Consumer:走 gRPC
👉 两个世界
3️⃣ 日志不一定可信,异常才是真相
text
日志打印 topic ≠ Client 实际拉取路由的 topic
八、最终 Checklist(上线前必查)
- NameServer 暴露 8081
- Broker 暴露 8080
- 使用 rocketmq-client-java 5.x
- 未配置
rocketmq.producer.topic - 所有 topic 已通过
mqadmin updatetopic创建 - Producer group 唯一
九、总结
RocketMQ 5.x 并不是"不能用",
而是 思维模型已经完全不同于 4.x。
真正的三大坑只有:
- gRPC 端口
- Topic 不自动创建
- Producer 默认 topic 与 send topic 混用
只要理解了这三点,
RocketMQ 5.x + Spring Boot 会非常稳定、清晰、现代化。
你这次踩的坑,非常值得被更多人看到。