zoukankan      html  css  js  c++  java
  • MQTT的使用介绍

    之前项目中使用到了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;
    }

    }
    //创建主题添加到集合中线程类
     
    一条路不能回头,就是一生要走许多路,有成长之路。很多事情不能自己掌控,即使再孤单再寂寞,仍要继续走下去,不许停也不能回头。------成长之旅。
  • 相关阅读:
    (转)运维角度浅谈MySQL数据库优化
    关于MySQL的null值
    MySQL优化——or条件优化
    MySQL优化原理
    Xcode Archive打包失败问题
    ionic3 对android包进行签名
    ios 审核未通过 相机相册权限问题
    js计算两个日期相差天数
    截取URL链接中字段的方法
    ionic3 自定义组件 滑动选择器 ion-multi-picker
  • 原文地址:https://www.cnblogs.com/java7115/p/9679894.html
Copyright © 2011-2022 走看看