RocketMQ 5.x + Spring Boot 发送消息失败全解析

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

真正的三大坑只有:

  1. gRPC 端口
  2. Topic 不自动创建
  3. Producer 默认 topic 与 send topic 混用

只要理解了这三点,

RocketMQ 5.x + Spring Boot 会非常稳定、清晰、现代化。


你这次踩的坑,非常值得被更多人看到

相关推荐
+VX:Fegn089512 小时前
计算机毕业设计|基于springboot + vue鲜花商城系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
识君啊13 小时前
SpringBoot 事务管理解析 - @Transactional 的正确用法与常见坑
java·数据库·spring boot·后端
CaracalTiger13 小时前
如何解决Unexpected token ‘<’, “<!doctype “… is not valid JSON 报错问题
java·开发语言·jvm·spring boot·python·spring cloud·json
苏渡苇14 小时前
Java + Redis + MySQL:工业时序数据缓存与持久化实战(适配高频采集场景)
java·spring boot·redis·后端·spring·缓存·架构
Hx_Ma1614 小时前
Springboot整合mybatis注解版
java·spring boot·mybatis
t***442315 小时前
Spring boot整合quartz方法
java·前端·spring boot
蛐蛐蜉蝣耶15 小时前
互联网大厂Java面试实录:当严肃面试官遇上搞笑程序员谢飞机
spring boot·微服务·java面试·电商系统·分布式系统·技术面试·程序员面试
enjoy嚣士16 小时前
springboot 之 时区问题
java·spring boot·后端·时区
沙河板混17 小时前
@RequestMapping的参数
java·spring boot·spring
indexsunny18 小时前
互联网大厂Java面试实战:Spring Boot到Kafka的技术问答解析
java·spring boot·redis·junit·kafka·spring security·microservices