开发 MQTT 客户端

admin / 开发 / ... / Reads: 3

前言

在java开发中,经常会用到消息服务,emq是我们现在通常使用的消息中间件,搭建一个emq服务器此处就不做过多讲解,本文主要说的是基于java环境搭建一个mqtt客户端。 你也可以参考emq官网的示例进行搭建,但官网过于简单,本文会有一些详细的参数配置 emq官方文档

1、引入pom依赖

<dependency>
    <groupId>org.eclipse.paho</groupId>
    <artifactId>org.eclipse.paho.client.mqttv3</artifactId>
    <version>1.2.0</version>
</dependency>
```

### 2、application.yml配置

```
# emq
mqtt:
  clientID: client-test
  username: admin
  password: public
  serverURI: tcp://127.0.0.1:1883
  isReconnect: true
  keepAliveInterval: 10
  topic: topic-test
```


### 3、ClientMQTT.java 相关代码
```
package cn.bdp.ws.emq;

import cn.bdp.ws.service.DmcMapService;
import cn.bdp.ws.service.MessageConsumeService;
import lombok.extern.slf4j.Slf4j;
import org.eclipse.paho.client.mqttv3.*;
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;

import javax.annotation.PostConstruct;
import javax.annotation.PreDestroy;
import java.util.concurrent.ScheduledExecutorService;

@Component
@Slf4j
public class ClientMQTT {

    @Value("${mqtt.serverURI}")
    private String serverURI;

    @Value("${mqtt.clientID}")
    private String clientId;

    @Value("${mqtt.username}")
    private String username;

    @Value("${mqtt.password}")
    private String password;

    @Value("${mqtt.isReconnect}")
    private boolean isReconnect;

    @Value("${mqtt.keepAliveInterval}")
    private int keepAliveInterval;

    @Value("${mqtt.topic}")
    private String mqttTopic;

    private MqttClient client;
    private MqttConnectOptions options;

    @Autowired
    DmcMapService dmcMapService;

    @Autowired
    MessageConsumeService messageConsumeService;

    @PostConstruct
    private void start() {

        try {
            client = new MqttClient(serverURI, clientId, new MemoryPersistence());
            options = new MqttConnectOptions();
            options.setCleanSession(true);
            options = new MqttConnectOptions();
            // 设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,这里设置为true表示每次连接到服务器都以新的身份连接
            options.setCleanSession(true);
            // 设置连接的用户名
            options.setUserName(username);
            // 设置连接的密码
            options.setPassword(password.toCharArray());
            // 设置超时时间 单位为秒
            options.setConnectionTimeout(10);
            // 设置会话心跳时间 单位为秒
            options.setKeepAliveInterval(keepAliveInterval);
            // 设置自动重连
            options.setAutomaticReconnect(true);
            System.out.println("Connecting to broker:" + serverURI);
            client.connect(options);
            System.out.println("Connected");
            System.out.println("Subscribe to topic:" + mqttTopic);
            client.subscribe(mqttTopic);
            // 此处使用的MqttCallbackExtended类而不是MqttCallback,是因为如果emq服务出现异常导致客户端断开连接后,重连后会自动调用connectComplete方法
            client.setCallback(new MqttCallbackExtended() {
                @Override
                public void connectComplete(boolean reconnect, String serverURI) {
                    System.out.println("连接完成...");
                    try {
                        // 重连后要自己重新订阅topic,这样emq服务发的消息才能重新接收到,不然的话,断开后客户端只是重新连接了服务,并没有自动订阅,导致接收不到消息
                        client.subscribe(mqttTopic);
                        log.info("订阅成功");
                    }catch (Exception e){
                        log.info("订阅出现异常:{}", e);
                    }

                }

                @Override
                public void connectionLost(Throwable cause) {
                    System.out.println("失去连接....");
                }

                @Override
                public void messageArrived(String topic, MqttMessage message) throws Exception {
                    // subscribe后得到的消息会执行到这里面
                    String content = new String(message.getPayload());
                    System.out.println("接收消息主题 : " + topic);
                    System.out.println("接收消息Qos : " + message.getQos());
                    System.out.println("接收消息内容 : " + content);
                    // 处理数据
                    messageConsumeService.dealEmqContent(content);
                }

                @Override
                public void deliveryComplete(IMqttDeliveryToken token) {
                    System.out.println("deliveryComplete....");
                }
            });

        } catch (Exception e) {
            e.printStackTrace();
        }
    }

    @PreDestroy
    public void end(){
        try {
            client.disconnect();
            client.close();
        } catch (MqttException e) {
            log.info("断开连接失败:{}", e);
        }
    }
}
```


此时,重启服务,如果emq服务器正常,客户端就可以正常连接
注意点(两个):

1、client.setCallback(new MqttCallbackExtended()
此处使用的MqttCallbackExtended类而不是MqttCallback,是因为如果emq服务出现异常导致客户端断开连接后,重连后会自动调用connectComplete方法

2、client.subscribe(mqttTopic)
重连后要自己重新订阅topic,这样emq服务发的消息才能重新接收到,不然的话,断开后客户端只是重新连接了服务,并没有自动订阅,导致接收不到消息


资料参考:
--------
https://docs.emqx.io/tutorial/v3/cn/client_dev/java.html


关于作者
--------
王硕,网名信平,十多年软件开发经验,业余架构师,精通Java/Python/Go等,喜欢研究技术,著有《PyQt 5 快速开发与实战》《Python 3.* 全栈开发》,多个业余开源项目托管在<a href='https://github.com/cxinping' target="_blank">GitHub</a>上,欢迎微博交流。

Comments

Make a comment

Author: admin

Publish at: ...

关于作者

王硕,网名信平,十多年软件开发经验,业余架构师,精通 C/Java/Python/Go 等,喜欢读书,音乐和宅在家里。
Email: xujieiata@163.com

www.ultrapower.com ultrapower 王硕的博客,专注于研究互联网产品和技术,提供中文精品教程。