MQTT协议的简单介绍和服务器的安装

转:http://blog.csdn.net/djun100/article/details/25752491

  最近公司做的项目中有用到消息推送,经过多方面的筛选之后确定了使用MQTT协议,相对于XMPP,MQTT更加轻量级,并且占用用户很少的带宽。

MQTT是IBM推出的一种针对移动终端设备的基于TCP/IP的发布/预订协议,可以连接大量的远程传感器和控制设备。

  MQTT的官网见:http://mqtt.org/。其中http://mqtt.org/software里面提供了官方推荐的各种服务器和客户端使用的各种语言版本的API。

  下面以服务器Apollo 1.6为例,之前尝试过使用ActiveMQ,效果很不理想,只能实现服务器和客户端一对一的通信,从官网上了解到Apollo属于activemq的一个子工程。先不管这些了,言归正传,以下在windows环境下。

  1、在这里下载Apollo服务器,下载后解压,然后运行apache-apollo-1.6\bin\apollo.cmd,输入create mybroker(名字任意取,这里是根据官网介绍的来取的)创建服务器实例,服务器实例包含了所有的配置,运行时数据等,并且和一个服务器进程关联。

  2、create mybroker之后会在bin目录下生成mybroker文件夹,里面包含有很多信息,其中etc\apollo.xml文件下是配置服务器信息的文件,etc\users.properties文件包含连接MQTT服务器时用到的用户名和密码,后面会介绍,可以修改原始的admin=password,可以接着换行添加新的用户名密码。

  3、打开cmd,运行…apache-apollo-1.6\bin\mybroker\bin\apollo-broker.cmd run 开启服务器,可以在浏览器中输入http://127.0.0.1:61680/查看是否安装成功,该界面展示了topic,连接数等很多信息。

  经过上面的简单步骤,服务器基本上就已经完成,下一篇将介绍Android客户端的编写和注意事项。

  客户端使用的API,开始我使用的是mqtt-client,使用过后发现问题百出,不能很好的满足要求,后来使用了官方推荐的Eclipse Paho,下面开始客户端代码的编写,为了方便测试这里有android和j2se两个工程:

  1、新建android工程MQTTClient

  2、MainActivity代码如下:

[java] view plaincopyprint?

package ldw.mqttclient;

import java.util.concurrent.Executors;

import java.util.concurrent.ScheduledExecutorService;

import java.util.concurrent.TimeUnit;

import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;

import org.eclipse.paho.client.mqttv3.MqttCallback;

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.MqttMessage;

import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;

import android.app.Activity;

import android.os.Bundle;

import android.os.Handler;

import android.os.Message;

import android.view.KeyEvent;

import android.widget.TextView;

import android.widget.Toast;

public class MainActivity extends Activity {

       private TextView resultTv;

       private String host = "tcp://127.0.0.1:1883";

       private String userName = "admin";

       private String passWord = "password";

       private Handler handler;

       private MqttClient client;

       private String myTopic = "test/topic";

       private MqttConnectOptions options;

       private ScheduledExecutorService scheduler;

       @Override

       protected void onCreate(Bundle savedInstanceState) {

              super.onCreate(savedInstanceState);

              setContentView(R.layout.main);

              resultTv = (TextView) findViewById(R.id.result);

              init();

              handler = new Handler() {

                     @Override

                     public void handleMessage(Message msg) {

                            super.handleMessage(msg);

                            if(msg.what == 1) {

                                   Toast.makeText(MainActivity.this, (String) msg.obj,

                                                 Toast.LENGTH_SHORT).show();

                                   System.out.println("-----------------------------");

                            } else if(msg.what == 2) {

                                   Toast.makeText(MainActivity.this, "连接成功", Toast.LENGTH_SHORT).show();

                                   try {

                                          client.subscribe(myTopic, 1);

                                   } catch (Exception e) {

                                          e.printStackTrace();

                                   }

                            } else if(msg.what == 3) {

                                   Toast.makeText(MainActivity.this, "连接失败,系统正在重连", Toast.LENGTH_SHORT).show();

                            }

                     }

              };

              startReconnect();

       }

       private void startReconnect() {

              scheduler = Executors.newSingleThreadScheduledExecutor();

              scheduler.scheduleAtFixedRate(new Runnable() {

                     @Override

                     public void run() {

                            if(!client.isConnected()) {

                                   connect();

                            }

                     }

              }, 0 * 1000, 10 * 1000, TimeUnit.MILLISECONDS);

       }

       private void init() {

              try {

                       //host为主机名,test为clientid即连接MQTT的客户端ID,一般以客户端唯一标识符表示,MemoryPersistence设置clientid的保存形式,默认为以内存保存

                     client = new MqttClient(host, "test",

                                   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 MqttCallback() {

                            @Override

                            public void connectionLost(Throwable cause) {

                                        //连接丢失后,一般在这里面进行重连

                                   System.out.println("connectionLost----------");

                            }

                            @Override

                            public void deliveryComplete(IMqttDeliveryToken token) {

                                        //publish后会执行到这里

                                   System.out.println("deliveryComplete---------"

                                                 + token.isComplete());

                            }

                            @Override

                            public void messageArrived(String topicName, MqttMessage message)

                                          throws Exception {

                                        //subscribe后得到的消息会执行到这里面

                                   System.out.println("messageArrived----------");

                                   Message msg = new Message();

                                   msg.what = 1;

                                   msg.obj = topicName+"---"+message.toString();

                                   handler.sendMessage(msg);

                            }

                     });

//                   connect();

              } catch (Exception e) {

                     e.printStackTrace();

              }

       }

       private void connect() {

              new Thread(new Runnable() {

                     @Override

                     public void run() {

                            try {

                                   client.connect(options);

                                   Message msg = new Message();

                                   msg.what = 2;

                                   handler.sendMessage(msg);

                            } catch (Exception e) {

                                   e.printStackTrace();

                                   Message msg = new Message();

                                   msg.what = 3;

                                   handler.sendMessage(msg);

                            }

                     }

              }).start();

       }

       @Override

       public boolean onKeyDown(int keyCode, KeyEvent event) {

              if(client != null && keyCode == KeyEvent.KEYCODE_BACK) {

                     try {

                            client.disconnect();

                     } catch (Exception e) {

                            e.printStackTrace();

                     }

              }

              return super.onKeyDown(keyCode, event);

       }

       @Override

       protected void onDestroy() {

              super.onDestroy();

              try {

                     scheduler.shutdown();

                     client.disconnect();

              } catch (MqttException e) {

                     e.printStackTrace();

              }

       }

}

  

  由于项目需要,我用到了心跳重连。根据这里的解释设置apollo.xml,主要有设置主机连接的地址。另外,options还有个setWill方法,如果项目中需要知道客户端是否掉线可以调用该方法。

  3、新建j2se工程MQTTServer

  4、Server代码如下:

[java] view plaincopyprint?

import java.awt.Container;

import java.awt.event.ActionEvent;

import java.awt.event.ActionListener;

import javax.swing.JButton;

import javax.swing.JFrame;

import javax.swing.JPanel;

import org.eclipse.paho.client.mqttv3.IMqttDeliveryToken;

import org.eclipse.paho.client.mqttv3.MqttCallback;

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.MqttMessage;

import org.eclipse.paho.client.mqttv3.MqttTopic;

import org.eclipse.paho.client.mqttv3.persist.MemoryPersistence;

public class Server extends JFrame {

       private static final long serialVersionUID = 1L;

       private JPanel panel;

       private JButton button;

       private MqttClient client;

       private String host = "tcp://127.0.0.1:1883";

//     private String host = "tcp://localhost:1883";

       private String userName = "test";

       private String passWord = "test";

       private MqttTopic topic;

       private MqttMessage message;

       private String myTopic = "test/topic";

       public Server() {

              try {

                     client = new MqttClient(host, "Server",

                                   new MemoryPersistence());

                     connect();

              } catch (Exception e) {

                     e.printStackTrace();

              }

              Container container = this.getContentPane();

              panel = new JPanel();

              button = new JButton("发布话题");

              button.addActionListener(new ActionListener() {

                     @Override

                     public void actionPerformed(ActionEvent ae) {

                            try {

                                   MqttDeliveryToken token = topic.publish(message);

                                   token.waitForCompletion();

                                   System.out.println(token.isComplete()+"========");

                            } catch (Exception e) {

                                   e.printStackTrace();

                            }

                     }

              });

              panel.add(button);

              container.add(panel, "North");

       }

       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 MqttCallback() {

                            @Override

                            public void connectionLost(Throwable cause) {

                                   System.out.println("connectionLost-----------");

                            }

                            @Override

                            public void deliveryComplete(IMqttDeliveryToken token) {

                                   System.out.println("deliveryComplete---------"+token.isComplete());

                            }

                            @Override

                            public void messageArrived(String topic, MqttMessage arg1)

                                          throws Exception {

                                   System.out.println("messageArrived----------");

                            }

                     });

                     topic = client.getTopic(myTopic);

                     message = new MqttMessage();

                     message.setQos(1);

                     message.setRetained(true);

                     System.out.println(message.isRetained()+"------ratained状态");

                     message.setPayload("eeeeeaaaaaawwwwww---".getBytes());

                     client.connect(options);

              } catch (Exception e) {

                     e.printStackTrace();

              }

       }

       public static void main(String[] args) {

              Server s = new Server();

              s.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

              s.setSize(600, 370);

              s.setLocationRelativeTo(null);

              s.setVisible(true);

       }

}

  上面代码跟客户端的代码差不多,这里就不做解释了。

  没什么好说的,MQTT就是这么简单,但开始在使用的时候要注意一些参数的设置来适应项目的需求。

  jar包下载地址:https://repo.eclipse.org/content/repositories/paho/org/eclipse/paho/mqtt-client/0.4.0/

  转自:http://www.longdw.com/mqtt-server-client-android/

时间: 2024-08-29 01:53:51

MQTT协议的简单介绍和服务器的安装的相关文章

Swift中协议的简单介绍

熟悉objective-c语言的同学们肯定对协议都不陌生,在Swift中苹果将 protocol 这种语法发扬的更加深入和彻底.Swift语言中的 protocol 不仅能定义方法还能定义属性,配合 extension 扩展的使用还能提供一些方法的默认实现,而且不仅类可以遵循协议,现在的枚举和结构体也能遵循协议了.基于此本文从 1,协议中定义属性和方法 , 2,协议的继承.聚合.关联类型 , 3,协议的扩展 , 4,Swift标准库中常见的协议 , 5,为什么要使用协议 5个方面结合自身的学习经

MQTT协议的简单了解

MQTT协议-MQTT协议解析(MQTT数据包结构) 协议就是通信双方的一个约定, 即,表示第1位传输的什么.第2位传输的什么--. 在MQTT协议中,一个MQTT数据包由: 固定头(Fixed header). 可变头(Variable header). 消息体(payload)三部分构成 MQTT 数据包结构 * 固定头(Fixed header),存在于所有MQTT数据包中,表示数据包类型及数据包的分组类标识 * 可变头(Variable header),存在于部分MQTT数据包中,数据包

HTTP协议的简单介绍

前传:HTTP协议的演变过程 HTTP(HyperText Transfer Protocol)协议是基于TCP的应用层协议,它不关心数据传输的细节,主要是用来规定客户端和服务端的数据传输格式,最初是用来向客户端传输HTML页面的内容.默认端口是80. 1.HTTP 0.9版本 1991年 这个版本就是最初用来向客户端传输HTML页面的,所以只有一个GET命令,然后服务器返回客户端一个HTML页面,不能是其他格式.利用这个版本完全可以构建一个简单的静态网站了. 2.HTTP 1.0版本 1996

一:AMQP协议标准简单介绍

一:AMQP协议?--->AMQP 是 Advanced Message Queuing Protocol,即高级消息队列协议.和前面罗列的技术不同,AMQP 是一个标准化的消息中间件协议--->她的理想是让不同语言,不同系统的应用互相通信,并提供一个简单统一的模型和编程接口.这样,人们就可以采用各种语言和平台来实现自己的应用,当需要和其他系统通信时,只要承认 AMQP 协议即可. 二:AMQP协议的形象化例子?--->世界各地的人们由于地理和历史的原因,使用着各种不同的语言,相互交流十

2015.12.22 类的三大特性 协议的简单介绍

类的三大特性:封装.继承.多态 封装: 1. 对于头文件里面的一些成员变量,可以使用@private @public @protected来限制权限 @public  公有的 类本身可以访问,子类可以继承,外部也可以访问 pxd->name @private 私有的 类本身可以访问,子类无法继承,外部不可以访问 @protected 受保护的 类本身可以访问, 子类可以继承, 外部不可以访问.默认就是protected 2.需要给外部使用的方法或者属性就定义在头文件里面声明.h 不需要暴露给外部

黑马程序员--Objective-C之--block的使用以及协议的简单介绍

块(block)是OC对ANSI C所做的扩展,使用块可以更好地简化OC编程,而且OC的很多API都依赖于块. 一.block (一).block的使用方法 语法格式如下: ^ [块返回值类型] (形参1,形参2,...) { } 必须以^开头,返回值类型通常都被省略,如果没有形参,建议使用void作为占位符. 如果程序需要多次调用已经定义的块,那么应该將块赋值给一个块变量,定义块变量的语法格式如下: block返回值类型 (^block变量名) (形参类型1,形参类型2,...); 定义块变量

一、tars简单介绍 二、tars 安装部署资料准备

1.github地址https://github.com/Tencent/Tars/ 2.tars是RPC开发框架,目前支持c++,java,nodejs,php 3.tars 在腾讯内部已经使用了快到十年,部署一万多台机器,在各种业务场景包括手机浏览器.应用宝.手机管家.手机游戏.手机qq等都有成功的实践. 4.Tars与业界其他同类或相识的应用框架相比:一是Tars提供了支持多语言(C++/Java)的高性能(性能可达40w/s)RPC开发框架,比如业界开源的Dubbo只支持Java,业界开

MQTT协议的使用

MQTT协议的使用: MQTT协议:是发布者,服务器(代理),客户(订阅者).之间通信的协议.客户通过MQTT协议,订阅自己的信息,订阅的内容给服务器.当需要有内容发布的时候,发布者将消息内容发送给服务器,服务器根据订阅了该消息的所有客户,将消息进行推送给所有订阅 该消息的客户(订阅者). MQTT协议在使用上,首先我们要将关于MQTT协议的代码弄到自己的工程中.这里只讲大致的使用过程,不确切的去讲. 首先要创建socket连接,这个和我们和网络通信的时候是一样的,首先都要去socket建立连接

Netty实现高性能IOT服务器(Groza)之手撕MQTT协议篇上

前言 诞生及优势 MQTT由Andy Stanford-Clark(IBM)和Arlen Nipper(Eurotech,现为Cirrus Link)于1999年开发,用于监测穿越沙漠的石油管道.目标是拥有一个带宽有效且使用很少电池电量的协议,因为这些设备是通过卫星链路连接的,当时这种设备非常昂贵. 与HTTP及其请求/响应范例相比,该协议使用发布/订阅体系结构.发布/订阅是事件驱动的,可以将消息推送到客户端.中央通信点是MQTT代理,它负责调度发送者和合法接收者之间的所有消息.向代理发布消息的