之前项目中使用到了mqtt,刚开始用着用着都不知道是干啥的,后来百度了一下:
-
MQTT
MQTT基于订阅者模型架构,客户端如果互相通信,必须在同一订阅主题下,即都订阅了同一个topic,客户端之间是没办法直接通讯的。订阅模型显而易见的好处是群发消息的话只需要发布到topic,所有订阅了这个topic的客户端就可以接收到消息了。
发送消息必须发送到某个topic,重点说明的是不管客户端是否订阅了该topic都可以向topic发送了消息,还有如果客户端订阅了该主题,那么自己发送的消息也会接收到。 -
MQTT特点
- 使用发布/订阅消息模式,提供一对多的消息发布,解除应用程序耦合。这一点很类似于XMPP,但是MQTT的信息冗余远小于XMPP
- 对负载内容屏蔽的消息传输
使用TCP/IP提供网络连接。主流的MQTT是基于TCP连接进行数据推送的,但是同样有基于UDP的版本,叫做MQTT-SN。这两种版本由于基于不同的连接方式,优缺点自然也就各有不同了 - 三种消息传输方式QoS:
- 0代表“至多一次”,消息发布完全依赖底层 TCP/IP 网络。会发生消息丢失或重复。这一级别可用于如下情况,环境传感器数据,丢失一次读记录无所谓,因为不久后还会有第二次发送。
- 1代表“至少一次”,确保消息到达,但消息重复可能会发生。
- 2代表“只有一次”,确保消息到达一次。这一级别可用于如下情况,在计费系统中,消息重复或丢失会导致不正确的结果。 (备注:由于服务端采用Mosca实现,Mosca目前只支持到QoS 1)
如果发送的是临时的消息,例如给某topic所有在线的设备发送一条消息,丢失的话也无所谓,0就可以了(客户端登录的时候要指明支持的QoS级别,同时发送消息的时候也要指明这条消息支持的QoS级别),如果需要客户端保证能接收消息,需要指定QoS为1,如果同时需要加入客户端不在线也要能接收到消息,那么客户端登录的时候要指定session的有效性,接收离线消息需要指定服务端要保留客户端的session状态。
使用流程:
1.导入pom文件:
<!-- MQTT -->
<dependency>
<groupId>org.eclipse.paho</groupId>
<artifactId>org.eclipse.paho.client.mqttv3</artifactId>
<version>1.1.1</version>
</dependency>
2.在web.xml配置监听器:
<!-- 配置Mqtt监听器监听器 -->
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<listener>
<listener-class>com.hx.lease.mqtt.MQTTServletContextListener</listener-class>
</listener>
3.编写mqtt的类:
public class MQTTServletContextListener implements ServletContextListener {
@Override
public void contextInitialized(ServletContextEvent servletContextEvent) {
System.out.println("--------------------------contextInitialized-------------------------------");
/**
* 开启8个主题的MQTT线程
*
* @throws ServletException
*/
ClientMQTT.openThread();
}
@Override
public void contextDestroyed(ServletContextEvent servletContextEvent) {
System.out.println("--------------------------------MQTTServlet.destroy()-------------------------");
}
}
//线程类:
public class ClientMQTT extends Thread {
public static final String HOST = "tcp://xxxxxx:1883";
private String TOPIC1 = ""; //订阅的主题为:连接
private static final String clientid = UUID.randomUUID().toString().replace("-", "");
private MqttClient client;
private MqttConnectOptions options;
private String userName = "saldjasl14kfc15jl985sjfi"; //非必须
private String passWord = "AADjv134,75sda"; //非必须
@Override
public void run() {
System.out.println(String.format("------------------------%s主题线程开启--------------------------------------", TOPIC1));
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(100);
// 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
options.setKeepAliveInterval(20);
// 设置回调
client.setCallback(new PushCallback());
MqttTopic topic = client.getTopic(TOPIC1);
//setWill方法,如果项目中需要知道客户端是否掉线可以调用该方法。设置最终端口的通知消息
//遗嘱 options.setWill(topic, "close".getBytes(), 2, true);
client.connect(options);
//订阅消息
int[] Qos = {1};
String[] topic1 = {TOPIC1};
client.subscribe(topic1, Qos);
} catch (Exception e) {
System.out.println(e.getMessage());
//e.printStackTrace();
}
}
public static void openThread() {
System.out.println("--------------------------MQTTServlet.init()--------------------");
/**
* 将主题添加到集合中
*/
List<String> topicList = new ArrayList<>();
topicList.add("connect");//订阅主题
topicList.add("disconnect");//断开连接
topicList.add("abnornaldisconnection");//非正常掉线连接
topicList.add("heartbeat");//心跳包
topicList.add("coinoperated");//
topicList.add("catchdoll");//抓到娃娃后上传
topicList.add("upset");//
topicList.add("bind");
/**
* 循环遍历主题集合
* 每个主题开启一个线程
*/
for (String topic : topicList) {
ClientMQTT clientMqttThread = new ClientMQTT();
clientMqttThread.setTOPIC1(topic);
clientMqttThread.start();
}
}
public String getTOPIC1() {
return TOPIC1;
}
public void setTOPIC1(String TOPIC1) {
this.TOPIC1 = TOPIC1;
}
}
//创建主题添加到集合中线程类