MQTT的使用介绍

妖精的绣舞 提交于 2019-11-28 20:33:18

之前项目中使用到了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;   }}
//创建主题添加到集合中线程类
 

 

易学教程内所有资源均来自网络或用户发布的内容,如有违反法律规定的内容欢迎反馈
该文章没有解决你所遇到的问题?点击提问,说说你的问题,让更多的人一起探讨吧!