之前项目中使用到了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; }}
//创建主题添加到集合中线程类
来源:https://www.cnblogs.com/java7115/p/9679894.html