MQTT的使用介绍

摘要:
MQTT特性使用发布/订阅消息模式来提供一对多消息发布和解耦应用程序。这与XMPP非常相似,但MQTT的信息冗余远远小于XMPP,XMPP使用TCP/IP为负载内容筛选的消息传输提供网络连接。主流的MQTT基于TCP连接进行数据推送,但也有一个基于UDP的版本,称为MQTTSN。

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

}
//创建主题添加到集合中线程类
 

免责声明:文章转载自《MQTT的使用介绍》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Unity碰撞消息(OnCollisionXXXX)和触发消息(OnTriggerXXXX)的调用情境linq日期查询下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

同样都是收费为啥丰巢热爆、蓝湖平静

简述:同样都是收费,为何面向大众的丰巢引发一波热潮,而面向专业设计师的蓝湖却没有什么大的波折。关于收费的一些市场情况和产品思考。 一、两者情况说明 1、丰巢         准备收费:2020年5月8日消息,4月30日起,丰巢快递柜开始实行超时收费的“会员制”,非会员包裹只可免费保存12小时,超过需收取0.5元/12小时的费用,收费引发广泛关注和质疑。就是...

ESA2GJK1DH1K升级篇: 阿里云物联网平台 OTA: 关于阿里云物联网平台 OTA 的升级流程说明

前言   鉴于有些用户直接想使用现成的物联网平台实现 OTA 远程升级   我就写一写这系列的文章   首先大家需要学习完这部分   https://www.cnblogs.com/yangfengwu/p/11828777.html    现在说一下具体流程 新增固件   一,选择物联网平台里面的          固件名称: 随意     所属产品:...

RabbitMQ 运转流程

生产者发送消息 1、生产者连接到 RabbitMQ Broker,建立一个连接(Connection),开启一个信道(Channel) 2、生产者声明一个交换器,并设置相关属性,比如交换机类型、是否持久化等 3、生产者声明一个队列并设置相关属性,比如是否排他、是否持久化、是否自动删除等 4、生产者通过路由键将交换器和队列绑定起来 5、生产者发送消息至 Ra...

Spring Boot 异步请求和异步调用,一文搞定

一、Spring Boot中异步请求的使用 1、异步请求与同步请求     特点: 可以先释放容器分配给请求的线程与相关资源,减轻系统负担,释放了容器所分配线程的请求,其响应将被延后,可以在耗时处理完成(例如长时间的运算)时再对客户端进行响应。 一句话:增加了服务器对客户端请求的吞吐量(实际生产上我们用的比较少,如果并发请求量很大的情况下,我们会通...

群发技术-使用python3给微信好友群发消息

本文介绍如何给个人微信好友群发消息。 微信个人号中的群发助手可以一次给30个发送消息,如果要给所有所有群发,则需要自己手动发送多次,或者借助程序实现了。本文即是程序实现教程 一、原理 在微信的官方网站上 https://weixin.qq.com/,提供了一个网页版的微信,地址为 https://wx.qq.com/ 登录该网页版微信后,可以在网页上给对方...

MQTT的$SYS主题定义

$SYS/broker/load/bytes/received:自代理启动以来已接收的字节总数。 $SYS/broker/load/bytes/sent:自代理启动以来发送的字节总数。 $SYS/broker/clients/connected:当前连接的客户端数 $SYS/broker/clients/disconnected:已在代理处注册但当前已断开...