前期准备
-
MQTT
MQTT(Message Queuing Telemetry Transport,消息队列遥测传输协议),是一种基于发布/订阅(publish/subscribe)模式的"轻量级"通讯协议,该协议构建于TCP/IP协议上,由IBM在1999年发布。MQTT最大优点在于,可以以极少的代码和有限的带宽,为连接远程设备提供实时可靠的消息服务。有以下三种不同质量的消息服务(QOS):
0:“至多一次”,消息发布完全依赖底层 TCP/IP 网络。会发生消息丢失或重复。这一级别可用于如下情况,环境传感器数据,丢失一次读记录无所谓,因为不久后还会有第二次发送。
1: “至少一次”,确保消息到达,但消息重复可能会发生。
2: “只有一次”,确保消息到达一次。这一级别可用于如下情况,在计费系统中,消息重复或丢失会导致不正确的结果 -
Mosquitto
Mosquitto是一个开源(BSD许可证)的消息代理,实现MQTT(消息队列遥测传输)协议。
点击网址,根据自己的电脑选择下载,安装
-
apache-Apollo服务器
点击 网址下载,解压即可
-
Paho客户端
点击网址下载,解压
简单入门
-
安装好mosquito后开启服务
-
服务器
添加环境变量,系统变量中添加 变量名为JAVA-HOME,变量值为E:\Java\jdk-8的系统变量,path中添加环境变量 %JAVA_HOME%\bin和 E:\Java\jdk-8。保存后重启电脑
解压后打开文件夹,在apache-apollo-1.71\bin的路径栏中输入cmd运行
在终端中输入apollo.cmd create xxxx,这里创建的账户是myserver
返回之前的文件夹可以看到多了一个myserver文件。打开它,在myserver文件的bin目录下的路径栏中输入cmd,进入终端。输入apollo-broker.cmd run启动服务器
在浏览器中输入http://127.0.0.1:61680,输入用户名admin,密码password进入服务器
如果能成功进入即表示MQTT服务器搭建成功(注意,不要关闭终端命令窗口)
注意事项
①账号和密码在users.properties文件中,可以自行修改
②在开启服务器的时候出现错误startup failed:java.lang…是因为jdk版本太高,Jdk9以后不包含类java.lang,8及其以下的版本即可。
③浏览器进入服务器时失败,可以看一下服务中的apollo-myserver服务是否开启
- 客户端
点击解压后的.exe文件运行,创建连接,输入相关的tcp,用户信息
服务器显示连接成功
- 订阅消息发布
在客户端创建两个连接,一个充当发布角色发布test主题,一个充当订阅者订阅test主题
发布test主题后,订阅者能收到发布者发布的test主题,服务器中也会显示
Java实例
在Java工程中导入jar依赖包,下载网址
- 订阅者
package mqtt;
import java.util.concurrent.ScheduledExecutorService;
import org.eclipse.paho.client.mqttv3.MqttClient;
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.eclipse.paho.client.mqttv3.MqttException;
import org.eclipse.paho.client.mqttv3.MqttTopic;
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;
public class client {
//tcp://MQTT安装的服务器地址:MQTT定义的端口号
public static final String HOST = "tcp://0.0.0.0:61613";
//定义一个主题
public static final String TOPIC = "test";
//定义MQTT的ID,可以在MQTT服务配置中指定
private static final String clientid = "c1";
private MqttClient client;
private MqttConnectOptions options;
private String userName = "admin";
private String passWord = "password";
private ScheduledExecutorService scheduler;
private void start() {
try {
client = new MqttClient(HOST, clientid, new MemoryPersistence());
// MQTT的连接设置
options = new MqttConnectOptions();
// 设置是否清空session,这里如果设置为false表示服务器会保留客户端的连接记录,这里设置为true表示每次连接到服务器都以新的身份连接
options.setCleanSession(true);
// 设置连接的用户名
options.setUserName(userName);
// 设置连接的密码
options.setPassword(passWord.toCharArray());
// 设置超时时间 单位为秒
options.setConnectionTimeout(10);
// 设置会话心跳时间 单位为秒 服务器会每隔1.5*20秒的时间向客户端发送个消息判断客户端是否在线,但这个方法并没有重连的机制
options.setKeepAliveInterval(20);
// 设置回调
client.setCallback(new PushCallback());
MqttTopic topic = client.getTopic(TOPIC);
client.connect(options);
//订阅消息
int[] Qos = {1};
String[] topic1 = {TOPIC};
client.subscribe(topic1, Qos);
} catch (Exception e) {
e.printStackTrace();
}
}
public static void main(String[] args) throws MqttException{
// TODO Auto-generated method stub
client client=new client();
client.start();
}
}
- 发布者
package mqtt;
import java.nio.charset.StandardCharsets;
import org.eclipse.paho.client.mqttv3.MqttClient;
import org.eclipse.paho.client.mqttv3.MqttConnectOptions;
import org.eclipse.paho.client.mqttv3.MqttDeliveryToken;
import org.eclipse.paho.client.mqttv3.MqttException;
import org.eclipse.paho.client.mqttv3.MqttMessage;
import org.eclipse.paho.client.mqttv3.MqttPersistenceException;
import org.eclipse.paho.client.mqttv3.MqttTopic;
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;
public class server {
public static final String HOST = "tcp://0.0.0.0:61613";
public static final String TOPIC = "test";//主题
private static final String clientid = "s2";//发布者
private MqttClient client;
private MqttTopic topic;
private String userName = "admin";
private String passWord = "password";
private MqttMessage message;
public server() throws MqttException {
// MemoryPersistence设置clientid的保存形式,默认为以内存保存
client = new MqttClient(HOST, clientid, new MemoryPersistence());
connect();
}
private void connect() {
MqttConnectOptions options = new MqttConnectOptions();
options.setCleanSession(false);
options.setUserName(userName);
options.setPassword(passWord.toCharArray());
// 设置超时时间
options.setConnectionTimeout(10);
// 设置会话心跳时间
options.setKeepAliveInterval(20);
try {
client.setCallback(new PushCallback());
client.connect(options);
topic = client.getTopic(TOPIC);
} catch (Exception e) {
e.printStackTrace();
}
}
public void publish(MqttTopic topic , MqttMessage message) throws MqttPersistenceException,
MqttException {
MqttDeliveryToken token = topic.publish(message);
token.waitForCompletion();
System.out.println("message is published completely!--- " + token.isComplete());//订阅信息已经发布
}
public static void main(String[] args) throws MqttException{
// TODO Auto-generated method stub
server server = new server();
server.message = new MqttMessage();
server.message.setQos(2);
server.message.setRetained(true);
server.message.setPayload("这是推送的内容-----test".getBytes());
server.publish(server.topic , server.message);
System.out.println( "ratained状态-------------------------"+server.message.isRetained() );
}
}
- 回调函数
package mqtt;
import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;
import org.eclipse.paho.client.mqttv3.MqttCallback;
import org.eclipse.paho.client.mqttv3.MqttMessage;
public class PushCallback implements MqttCallback{
public void connectionLost(Throwable cause) {
// 连接丢失后,一般在这里面进行重连
System.out.println("连接断开,可以做重连");
}
public void deliveryComplete(IMqttDeliveryToken token) {
System.out.println("deliveryComplete--------------------" + token.isComplete()); //传递推送消息成功
}
public void messageArrived(String topic, MqttMessage message) throws Exception {
// subscribe后得到的消息会执行到这里面
System.out.println("接收消息主题 : " + topic);
System.out.println("接收消息Qos : " + message.getQos());
System.out.println("接收消息内容 : " + new String(message.getPayload()));
}
}
- 运行结果
两个订阅者,一个发布者,主题为test1
浏览器服务端
总结+参考
MQTT还值得深入学习。代码借鉴的别人的,编程这方面还要努力
MQTT入门
Java编写mqtt客户端