最近有需求要了解一下各个推送的协议,目前了解到实现推送的三个主要方式:MQTT、XMPP和Google Cloud Message(GCM)。第三种方式暂不研究,前两种都要看一看,本篇讨论一下MQTT协议吧。本文使用阿里云Ubuntu云服务器安装代理服务器,使用eclipse paho实现的MqttClient编写代码。文中的所使用的帐户名和密码在本文发布后将会更改,请各位自行搭建环境。本文包括如下内容:java
MQTT全称是Message Queuing Telemetry Transport,MQTT是IBM开发的基于TCP/IP协议的轻量级通信协议。MQTT是一个客户端服务端架构的发布-订阅(publish-subscribe)的消息传输协议。它的设计思想是开放、简单、轻量、易于实现。这些特色使它适用于受限环境。例如,但不只限于:android
做为一个物联网专业的毕业生,看了以上的描述已经心动了,很适合做为传感器节点之间的通信协议哇!哦,忘了,我如今是个Androider……MQTT控制报文头部仅有2字节的长度,下降了网络传输所须要的流量。MQTT支持三种不一样级别的服务质量(Quality of Service,QoS)为不一样场景提供消息可靠性:git
若是各位读完了这些仍然以为不过瘾,没有戳中各位的痛点,能够去读一下MQTT的协议规范,这里中英文版本都有挑本身爱看的读一下就好。github
首先须要一个代理服务器,这里mqtt代理服务器使用的是apache的apollo,apollo支持STOMP,AMQP,MQTT,Openwire,SSL和WebSockets。下载戳这。apache
下载到本地以后,将之上传到服务器上:bash
$ scp 文件名 $username@$ip:~
复制代码
解压tar.gz:服务器
$ tar zxvf apache-apollo-1.7.1-unix-distro.tar.gz
复制代码
进入解压后的bin目录下执行apollo create testbroker命令建立一个名称为testbroker的代理服务器。网络
$ cd apache-apollo-1.7.1-unix-distro.tar.gz/bin/
$ ./apollo create testbroker
复制代码
输入ls命令就能够看到文件夹下多了一个testbroker的文件夹session
该文件夹下的apollo.xml中配置了端口和ip,不过这里就无论了。代理服务器配置完毕,接下来就是下载paho实现的mqtt client的jar包了。 下载地址架构
这里利用Idea编写Java程序实现,对于Android程序来讲只须要稍加修改就可直接使用。首先新建一个Java项目,接着将上面下载的jar包做为依赖导入。首先编写服务端:
import org.eclipse.paho.client.mqttv3.*;
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;
public class MqttServer {
/** * 代理服务器ip地址 */
public static final String MQTT_BROKER_HOST = "tcp://xiasuhuei321.com:61613";
/** * 订阅标识 */
public static final String MQTT_TOPIC = "test";
private static String userName = "admin";
private static String password = "password";
/** * 客户端惟一标识 */
public static final String MQTT_CLIENT_ID = "android_server_xiasuhuei321";
private static MqttTopic topic;
private static MqttClient client;
public static void main(String... args) {
// 推送消息
MqttMessage message = new MqttMessage();
try {
client = new MqttClient(MQTT_BROKER_HOST, MQTT_CLIENT_ID, new MemoryPersistence());
MqttConnectOptions options = new MqttConnectOptions();
options.setCleanSession(true);
options.setUserName(userName);
options.setPassword(password.toCharArray());
options.setConnectionTimeout(10);
options.setKeepAliveInterval(20);
topic = client.getTopic(MQTT_TOPIC);
message.setQos(1);
message.setRetained(false);
message.setPayload("message from server".getBytes());
client.connect(options);
while (true) {
MqttDeliveryToken token = topic.publish(message);
token.waitForCompletion();
System.out.println("已经发送");
Thread.sleep(10000);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
复制代码
这里的逻辑很是简单,建立一个MqttClient,每十秒发送一次消息,订阅了相应topic的客户端将会收到这个消息。接下来编写客户端:
import org.eclipse.paho.client.mqttv3.*;
import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;
public class MyMqttClient {
/** * 代理服务器ip地址 */
public static final String MQTT_BROKER_HOST = "tcp://xiasuhuei321.com:61613";
/** * 客户端惟一标识 */
public static final String MQTT_CLIENT_ID = "android_xiasuhuei321";
/** * 订阅标识 */
public static final String MQTT_TOPIC = "xiasuhuei321";
/** * */
public static final String USERNAME = "admin";
/** * 密码 */
public static final String PASSWORD = "password";
private volatile static MqttClient mqttClient;
private static MqttConnectOptions options;
public static void main(String... args) {
try {
// host为主机名,clientid即链接MQTT的客户端ID,通常以客户端惟一标识符表示,
// MemoryPersistence设置clientid的保存形式,默认为之内存保存
// 设备id不要太骚气!!!!!!!
mqttClient = new MqttClient(MQTT_BROKER_HOST, MQTT_CLIENT_ID, new MemoryPersistence());
// 配置参数信息
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);
// 链接
mqttClient.connect(options);
// 订阅
mqttClient.subscribe("test");
// 设置回调
mqttClient.setCallback(new MqttCallback() {
@Override
public void connectionLost(Throwable throwable) {
System.out.println("connectionLost");
}
@Override
public void messageArrived(String s, MqttMessage mqttMessage) throws Exception {
System.out.println("Topic: " + s + " Message: " + mqttMessage.toString());
}
@Override
public void deliveryComplete(IMqttDeliveryToken iMqttDeliveryToken) {
}
});
} catch (Exception e) {
e.printStackTrace();
}
}
}
复制代码
接下来启动服务端和客户端看一下效果
到这也差很少了,说实话,在Android中难的历来都不是实现推送,而是如何保证接收推送的服务存活。在Android对后台服务限制愈来愈大的如今,本身实现推送的意义可能并非很是大。可是对于一些特殊的应用场景下,好比用户打开应用进行的一些操做须要用到长链接,本身实现推送可能会更加可靠一些(听朋友说三方推送有时会莫名其妙收不到推送)。