前言
为了不浪费大家的时间请看这里:
- Java中如何建立MQTT连接;
- 如何通过订阅MQTT主题来发送消息以及接收消息;
- 如何在回调函数中调用我们的service服务存储数据;
- 如何搭建MQTT服务器;
Java中建立MQTT连接
在pom.xml添加如下设置
<dependency>
<groupId>org.eclipse.paho</groupId>
<artifactId>org.eclipse.paho.client.mqttv3</artifactId>
<version>1.2.2</version>
</dependency>
客户端的基础使用
public class MyMqttClient {
public static final String HOST = "tcp://ip:port";
public static final String TOPIC1 = "/World";
private static final String clientid = "88888888";
private MqttClient client;
private MqttConnectOptions options;
private String userName = ""; // 非必须
private String passWord = ""; // 非必须
// 开始连接
public void start() {
try {
// host为主机名,clientid即连接MQTT的客户端ID,一般以唯一标识符表示,MemoryPersistence设置clientid的保存形式,默认为以内存保存
client = new MqttClient(HOST, clientid, new MemoryPersistence());
// MQTT的连接设置
options = new MqttConnectOptions();
// 设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,设置为true表示每次连接到服务器都以新的身份连接
options.setCleanSession(false);
// 设置连接的用户名
options.setUserName(userName);
// 设置连接的密码
options.setPassword(passWord.toCharArray());
// 设置超时时间 单位为秒
options.setConnectionTimeout(10);
// 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
options.setKeepAliveInterval(20);
// 设置断开后重新连接
options.setAutomaticReconnect(true);
// 设置回调
client.setCallback(new PushCallback());
MqttTopic topic = client.getTopic(TOPIC1);
// setWill方法,如果项目中需要知道客户端是否掉线可以调用该方法。设置最终端口的通知消息 遗嘱
options.setWill(topic, "close".getBytes(), 1, true);
client.connect(options);
// 订阅消息
sub(TOPIC1);
} catch (Exception e) {
e.printStackTrace();
}
}
/**
* 关闭MQTT连接
*/
public void close() throws MqttException {
client.close();
client.disconnect();
}
/**
* 向某个主题发布消息 默认qos:1
*
* @param topic:发布的主题
* @param msg:发布的消息
*/
public void pub(String topic, String msg) throws MqttException {
MqttMessage mqttMessage = new MqttMessage();
//mqttMessage.setQos(2);
mqttMessage.setPayload(msg.getBytes());
MqttTopic mqttTopic = client.getTopic(topic);
MqttDeliveryToken token = mqttTopic.publish(mqttMessage);
token.waitForCompletion();
}
/**
* 向某个主题发布消息
*
* @param topic: 发布的主题
* @param msg: 发布的消息
* @param qos: 消息质量 Qos:0、1、2
*/
public void pub(String topic, String msg, int qos) throws MqttException {
MqttMessage mqttMessage = new MqttMessage();
mqttMessage.setQos(qos);
mqttMessage.setPayload(msg.getBytes());
MqttTopic mqttTopic = client.getTopic(topic);
MqttDeliveryToken token = mqttTopic.publish(mqttMessage);
token.waitForCompletion();
}
/**
* 订阅某一个主题 ,此方法默认的的Qos等级为:1
*
* @param topic 主题
*/
public void sub(String topic) throws MqttException {
client.subscribe(topic);
}
/**
* 订阅某一个主题,可携带Qos
*
* @param topic 所要订阅的主题
* @param qos 消息质量:0、1、2
*/
public void sub(String topic, int qos) throws MqttException {
client.subscribe(topic, qos);
}
/**
* main函数自己测试用
*/
public static void main(String[] args) throws MqttException {
MyMqttClient mqttConnect = new MyMqttClient();
mqttConnect.start()
}
回调函数
public class PushCallback extends BaseController implements MqttCallback {
public void connectionLost(Throwable cause) {
System.out.println("connectionLost---------连接断开,可以做重连");
}
public void deliveryComplete(IMqttDeliveryToken token) {
System.out.println("deliveryComplete---------" + token.isComplete());
}
public void messageArrived(String topic, MqttMessage message) throws Exception {
System.out.println("接收消息主题 : " + topic);
System.out.println("接收消息Qos : " + message.getQos());
String res = new String(message.getPayload());
System.out.println("接收消息内容 : " + res);
//通过上下文的方式获取Service,然后在这个地方保存数据即可
}
}
我的问题
不能直接在回调方法中使用注解形式来获取对象和Service(如@Autowired、@Service),直接使用注解会报java.lang.NullException错误,然后断开MQTT连接。
我的解决方法
通过上下文的方式获取到自己的Service服务,剩下的就跟直接在Controller 中操作一样了。
获取上下文
@Component
public class SpringUtil extends ApplicationObjectSupport {
public static ApplicationContext context;
public static Object getBean(String serviceName){
return context.getBean(serviceName);
}
@Override
protected void initApplicationContext(ApplicationContext context) throws BeansException {
super.initApplicationContext(context);
SpringUtil.context = context;
}
}
获取Service
ApplicationContext context = SpringUtil.context; //获取Spring容器
MyService s = context.getBean(MyService.class); //获取bean
文章内容太少怎么办
内容太少,这里贴一下阿里云MQTT服务器的安装过程。
1.下载安装包
wget https://packages.emqx.io/emqx-ce/v2.3.10/emqttd-centos7-v2.3.10.zip
2.安装解压压缩包 软件
yum install unzip -y
3.解压服务器的压缩包
unzip emqttd-centos7-v2.3.10.zip
4.运行MQTT软件
4.1、进入 emqttd 主文件
cd emqttd
4.2、开始安装 emqttd 主文件 ,注意有小数点
./bin/emqttd console
出现如下图,CTRL+c 关闭控制台
4.3、启动守护进程模式
./bin/emqttd start
4.4、正常运行状态,查询命令返回
./bin/emqttd_ctl status
安装完成怎么办?配置安全组打开相应的端口,然后在浏览器访问(http://ip:port)
http://127.0.0:18083
默认登录帐号:admin
默认登录密码:public
注:其他端口先不说,18083必须先打开
MQTT 与HTTP的区别
MQTT 是基于 发布(Publish)/订阅(Subscribe) 模式来进行通信及数据交换的,与 HTTP 的 请求(Request)/应答(Response) 的模式有本质的不同。
订阅者(Subscriber) 会向 消息服务器(Broker) 订阅一个 主题(Topic) 。成功订阅后,消息服务器会将该主题下的消息转发给所有的订阅者。