mqtt消费堆积

mqtt发送主题成功,但消费主题只有一个

代码实例:

java 复制代码
    @ServiceActivator(inputChannel = "caGetConfig")
    @Override
    public void handleMessage(Message<?> message) throws MessagingException {
        try {
            String payload = (String) message.getPayload();
            String topic = (String) message.getHeaders().get(MQTTRECEIVEDTOPIC);
            log.info("接收到MQTT消息,主题: {}, 消息内容: {}", topic, payload);
            try {
                if (StringUtils.isEmpty(payload)) {
                    log.info("接收到空的MQTT消息,主题: {}", topic);
                    CompletableFuture.runAsync(() -> generateVideoService.getNeMessage());
                    return;
                }
                if (StringUtils.isBlank(topic)) {
                    log.info("接收到空的MQTT消息,主题: {}", topic);
                    CompletableFuture.runAsync(() -> generateVideoService.getNeMessage());
                }
                // 处理MQTT消息
                List<String> onlineStatusNewLis = JSON.parseArray(payload, String.class);
                if (CollectionUtils.isNotEmpty(onlineStatusNewLis)){
                    generateVideoService.getNeMessageByUrl(onlineStatusNewLis)
                }
            } catch (Exception e) {
                log.error("分发MQTT消息失败,主题: {}, 消息: {}", topic, payload, e);
            }
        } catch (Exception e) {
            log.error("处理MQTT消息失败,消息: {}", message, e);
        }
    }

问题分析:

1. MQTT 消费者是单线程的

@ServiceActivator(inputChannel = "caGetConfig") 基于 Spring Integration MQTT,其 inputChannel 默认是 PublishSubscribeChannel(同步调用),即消息的接收和处理在同一线程中串行执行。只有当 handleMessage() 方法返回后,才能处理下一条消息。

2. 使用 CompletableFuture.runAsync 异步调用

generateVideoService.getNeMessageByUrl(onlineStatusNewLis);方法执行完才能返回。如果 onlineStatusNewLis 列表较大,或者数据库查询较慢,这段时间内 MQTT 消费线程就会被阻塞。

如果该方法里面是一个 while(!Thread.currentThread().isInterrupted()) 的无限循环 ,这个循环会在 MQTT 消费线程上执行,永远无法返回,后续所有消息都将被彻底阻塞。

添加异步编排执行,解决阻塞

java 复制代码
// 处理MQTT消息
List<String> onlineStatusNewLis = JSON.parseArray(payload, String.class);
if (CollectionUtils.isNotEmpty(onlineStatusNewLis)){
  CompletableFuture.runAsync(() -> generateVideoService.getNeMessageByUrl(onlineStatusNewLis));
                    
}
相关推荐
掌心向暖RPA自动化12 小时前
如何获取网页某个元素在屏幕可见部分的中心坐标影刀RPA懒加载坐标定位技巧
java·javascript·自动化·rpa·影刀rpa
日取其半万世不竭12 小时前
Minecraft Java版社区服务器搭建教程(Linux,适合新手)
java·linux·服务器
MengMeng_102312 小时前
win10 蓝牙连接音响没有声音设备选项
windows
TeamDev13 小时前
JxBrowser 9.0.0 版本发布啦!
java·前端·混合应用·jxbrowser·浏览器控件·跨平台渲染·原声输入
AI人工智能+电脑小能手13 小时前
【大白话说Java面试题】【Java基础篇】第24题:Java面向对象有哪些特征
java·开发语言·后端·面试
强殖装甲凯普13 小时前
处理Windows没有msi的默认打开方式
windows·安装·msi
mOok ONSC13 小时前
mysql9.0windows安装
windows·adb
AI人工智能+电脑小能手14 小时前
【大白话说Java面试题】【Java基础篇】第25题:JDK1.8的新特性有哪些
java·开发语言·后端·面试
likerhood14 小时前
SLF4J: Failed to load class “StaticLoggerBinder“ 解决
java·log4j·maven
早日退休!!!14 小时前
大模型推理瓶颈七层分析模型
java·服务器·数据库