SpringBoot实现MQTT订阅以及发布

4 篇文章 1 订阅
2 篇文章 0 订阅
import lombok.Data;

import java.io.Serializable;

/**
 * 基础MQTT配置
 */
@Data
public class BaseMqttConfig implements Serializable {


    private static final long serialVersionUID = -4049488718513019015L;

    /**
     * 是否开启
     */
    public Boolean enable = false;

    /**
     * 订阅地址可设置多个
     */
    public String hostUrls;

    /**
     * MQTT账号
     */
    public String hostName;

    /**
     * MQTT密码
     */
    public String password;

    /**
     * 客户端ID
     */
    public String clientId;

    /**
     * 主题
     */
    public String topics;

    /**
     * 超时时间 单位为秒
     */
    public Integer timeout = 10;

    /**
     * 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送心跳判断客户端是否在线
     */
    public Integer keepAlive = 30;

    /**
     * 清除会话(设置为false,断开连接,重连后使用原来的会话 保留订阅的主题,能接收离线期间的消息)
     */
    public Boolean clearSession = false;

    /**
     * “遗嘱”消息的话题
     */
    public String willTopic = "offline";

}
power:
  mqtt:
    charge:
      enable: true
      hostUrls: tcp://test.test.com:2354
      hostName: test
      password: test
      clientId: mqtt_server
      topics: test/power/n/up/#
      timeout: 10
      keepAlive: 60
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.springframework.beans.factory.annotation.Qualifier;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.integration.channel.ExecutorChannel;
import org.springframework.integration.dsl.IntegrationFlow;
import org.springframework.integration.dsl.IntegrationFlows;
import org.springframework.integration.mqtt.core.DefaultMqttPahoClientFactory;
import org.springframework.integration.mqtt.core.MqttPahoClientFactory;
import org.springframework.integration.mqtt.inbound.MqttPahoMessageDrivenChannelAdapter;
import org.springframework.integration.mqtt.outbound.MqttPahoMessageHandler;
import org.springframework.integration.mqtt.support.DefaultPahoMessageConverter;
import org.springframework.messaging.MessageChannel;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;

import javax.annotation.Resource;

@Configuration
@ConditionalOnProperty(name = "power.mqtt.charge.enable", havingValue = "true")
public class PowerMqttServer extends BaseMqttServer {

    @Resource
    private PowerMqttConfig config;

    @Resource
    @Qualifier(ThreadPoolConfig.COMMON_THREAD_POOL)
    private ThreadPoolTaskExecutor executor;

    @Resource
    private PowerMessageHandler messageHandle;

    /**
     * 客户端工厂
     *
     * @return
     */
    @Bean
    public MqttPahoClientFactory initializeMqttFactory() {
        try {
            DefaultMqttPahoClientFactory factory = new DefaultMqttPahoClientFactory();
            MqttConnectOptions options = new MqttConnectOptions();
            options.setServerURIs(config.hostUrls.split(",")); // 设置代理端的URL地址,可以是多个
            options.setUserName(config.hostName); // 设置连接的用户名
            options.setPassword(config.password.toCharArray());  // 设置连接的密码
            options.setConnectionTimeout(config.timeout);// 设置超时时间 单位为秒
            options.setKeepAliveInterval(config.keepAlive);// 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送心跳判断客户端是否在线 但这个方法并没有重连的机制
            options.setWill("willTopic", config.willTopic.getBytes(), 2, false);// 设置“遗嘱”消息的话题,若客户端与服务器之间的连接意外中断,服务器将发布客户端的“遗嘱”消息。
            factory.setConnectionOptions(options);
            AssertLog.info("\n\n\t\t\t\t\t\t\t\t======hostUrl:{}======MQTT已建立连接,hostName:{}======\n", config.hostUrls, config.hostName);
            return factory;
        } catch (Exception e) {
            AssertLog.error("=======MQTT建立失败======", e.getCause());
        }
        return new DefaultMqttPahoClientFactory();
    }

    /**
     * 入站消息管道
     *
     * @return
     */
    @Bean
    public MessageChannel mqttInboundChannel() {
        AssertLog.info("==============MQTT入站消息管道加载成功==============");
        return new ExecutorChannel(executor);// 用线程池
    }

    /**
     * Mqtt 管道适配器
     *
     * @return
     */
    @Bean
    public MqttPahoMessageDrivenChannelAdapter adapter(MqttPahoClientFactory factory) {
        String inClientId = config.clientId + MqttConstant.CLIENT_NAME_IN;
        MqttPahoMessageDrivenChannelAdapter adapter = new MqttPahoMessageDrivenChannelAdapter(inClientId, factory, config.topics.split(","));
        adapter.setCompletionTimeout(5000);
        adapter.setQos(2);
        DefaultPahoMessageConverter converter = new DefaultPahoMessageConverter();
        converter.setPayloadAsBytes(true);
        adapter.setConverter(converter);
        return adapter;
    }

    /**
     * 消息消费者 (接收,处理来自mqtt的消息)
     *
     * @param adapter
     * @return
     */
    @Bean
    public IntegrationFlow mqttInbound(MqttPahoMessageDrivenChannelAdapter adapter) {
        return IntegrationFlows.from(adapter)
                .channel(new ExecutorChannel(executor))
                .handle(messageHandle)
                .get();
    }

    /**
     * 出站处理器 (向 mqtt 发送消息 生产者)
     *
     * @param factory
     * @return
     */
    @Bean
    public IntegrationFlow mqttOutboundFlow(MqttPahoClientFactory factory) {
        AssertLog.info("==============MQTT出站消息管道加载成功==============");
        String outClientId = config.clientId + MqttConstant.CLIENT_NAME_OUT;
        MqttPahoMessageHandler handler = new MqttPahoMessageHandler(outClientId, factory);
        handler.setAsync(true);  // 如果设置成true,即异步,发送消息时将不会阻塞。
        DefaultPahoMessageConverter converter = new DefaultPahoMessageConverter();//设置转换器 发送bytes数据
        converter.setPayloadAsBytes(true);
        handler.setDefaultTopic(config.topics.split(",")[0]);
        return IntegrationFlows.from(MqttConstant.CHANNEL_NAME_OUT).handle(handler).get();
    }

}
@Component
public class PowerMessageHandler implements MessageHandler {


    @Override
    public void handleMessage(Message<?> message) throws MessagingException {
        String topic = (String) message.getHeaders().get(MqttHeaders.RECEIVED_TOPIC);
       
    }
}
import org.springframework.integration.annotation.MessagingGateway;
import org.springframework.integration.mqtt.support.MqttHeaders;
import org.springframework.messaging.handler.annotation.Header;
import org.springframework.stereotype.Component;

@Component
@MessagingGateway(defaultRequestChannel = MqttConstant.CHANNEL_NAME_OUT)
public interface MqttSendService {

    /**
     * 指定topic进行消息发送
     *
     * @param topic
     * @param payload
     */
    void sendToMqtt(@Header(MqttHeaders.TOPIC) String topic, String payload);

    /**
     * 指定topic进行消息发送
     *
     * @param topic
     * @param qos
     * @param payload
     */
    void sendToMqtt(@Header(MqttHeaders.TOPIC) String topic, @Header(MqttHeaders.QOS) Integer qos, String payload);

    /**
     * 指定topic进行消息发送
     *
     * @param topic
     * @param qos
     * @param payload
     */
    void sendToMqtt(@Header(MqttHeaders.TOPIC) String topic, @Header(MqttHeaders.QOS) Integer qos, byte[] payload);

}
import java.lang.management.ManagementFactory;

/**
 * MQTT基本参数
 */
public class MqttConstant {

    /**
     * 出站-生产者
     */
    public static final String CHANNEL_NAME_OUT = "mqttOutboundChannel";

    /**
     * 入站-消费者
     */
    public static final String CHANNEL_NAME_IN = "mqttInputChannel";

    /**
     * 出站客户端ID
     */
    public static final String CLIENT_NAME_OUT = "_publish_" + ManagementFactory.getRuntimeMXBean().getName();
    ;

    /**
     * 入站客户端ID
     */
    public static final String CLIENT_NAME_IN = "_subscribe_" + ManagementFactory.getRuntimeMXBean().getName();

}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
Spring Boot 是一个用于快速构建 Spring 应用程序的框架,而 MQTT 是一种轻量级的消息传递协议。将 Spring BootMQTT 集成可以让我们更加容易地创建一个可靠的、实时的通信系统。以下是如何将 Spring Boot 集成 MQTT 发布/订阅的步骤: 1. 引入 MQTT 相关依赖:在 pom.xml 文件中添加以下依赖: <dependency> <groupId>org.springframework.integration</groupId> <artifactId>spring-integration-mqtt</artifactId> <version>1.1.3.RELEASE</version> </dependency> 2. 配置 MQTT:在 application.properties 文件中添加以下配置: # MQTT 配置 mqtt.client.id=spring-boot-mqtt mqtt.username=username mqtt.password=password mqtt.url=tcp://localhost:1883 3. 编写发布代码:使用 Spring Integration 的 PublishSubscribeChannel 和 MqttPahoMessageHandler 来向 MQTT 发布消息。 @Autowired private MessageChannel mqttOutboundChannel; public void sendToMqtt(String message) { mqttOutboundChannel.send(MessageBuilder.withPayload(message).build()); } 4. 编写订阅代码:使用 Spring Integration 的 MqttPahoMessageDrivenChannelAdapter 和 MessageHandler 来实现订阅。 @Bean public MessageChannel mqttInputChannel() { return new DirectChannel(); } @Bean public MqttPahoMessageDrivenChannelAdapter mqttInbound() { String clientId = MqttAsyncClient.generateClientId(); MqttPahoMessageDrivenChannelAdapter adapter = new MqttPahoMessageDrivenChannelAdapter(mqtt.url, clientId, mqtt.topic); adapter.setCompletionTimeout(5000); adapter.setConverter(new DefaultPahoMessageConverter()); adapter.setQos(1); adapter.setOutputChannel(mqttInputChannel()); return adapter; } @Bean @ServiceActivator(inputChannel = "mqttInputChannel") public MessageHandler handler() { return message -> { String payload = message.getPayload().toString(); System.out.println("MQTT Received: " + payload); }; } 通过以上步骤,我们可以轻松地集成 MQTT 发布/订阅功能。注意,在实际的应用程序中,我们需要为 MQTT 客户端定义一个独特的客户端 ID,并在订阅消息时指定选择的 MQTT 主题。这可以确保不同的客户端能够接收到正确的消息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

NuoleAQ

你的鼓励是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值