MQTT开发总结
首先,我们简单的提一下mqtt是什么;MQTT(Message Queuing TelemetryTransport,消息队列遥测传输)是IBM开发的一个即时通讯协议,有可能成为物联网的重要组成部分。该协议支持所有平台,几乎可以把所有联网物品和外部连接起来,被用来当做传感器和致动器(比如通过Twitter让房屋联网)的通信协议。
上面的这些语言都太官方了,我觉得总结一句话就是:连上mqtt之后,服务器那边就可以在你不断的情况下给你推送数据,并且实时性比较好;
现在就让我们来总结一下我们所使用的mqtt到底是怎么工作的,方便以后查阅的时候少走弯路。
流程图:
一、既然要使用官方提供的mqtt协议,那么就必须导入官方给出的jar包我们使用的是IBM公司提供的jar包。
二、既然要让它一直在后台运行,则必须通过服务来是这个操作符合我们的要求,为啥用服务这个就不需要我多说了啥。
既然是链接前后台的服务器,那么链接的状态可能要分很多种,例如:
[java] view plain copy
print?
1. switch (connectionStatus)
2. {
3. case INITIAL:
4. status = \"Please wait\";
5. break;
6. case CONNECTING:
7. status = \"Connecting...\";
8. break;
9. case CONNECTED:
10. status = \"Connected\";
11. break;
12. case NOTCONNECTED_UNKNOWNREASON:
13. status = \"Not connected - waiting for network connection\";
14. break;
15. case NOTCONNECTED_USERDISCONNECT:
16. status = \"Disconnected\";
17. break;
18. case NOTCONNECTED_DATADISABLED:
19. status = \"Not connected - background data disabled\";
20. break;
21. case NOTCONNECTED_WAITINGFORINTERNET:
22. status = \"Unable to connect\";// 无线网断开之后就提示这个信息
23. break;
24. }
就会有以上的六种状态信息;链接的状态是通过发送广播的形式发送的:
(1)、断开链接的广播:
[html] view plain copy
print?
1. connectionStatus = MQTTConnectionStatus.NOTCONNECTED_USERDISCO
NNECT;
2. // inform the app that the app has successfully disconnected
3. broadcastServiceStatus(\"Disconnected\");
(2)、尝试链接,但未链接上:
[html] view plain copy
print?
1. connectionStatus = MQTTConnectionStatus.NOTCONNECTED_WAITINGFORINTERNET;
2. // inform the app that we are not connected any more
3. broadcastServiceStatus(\"Connection lost - no network connection\");
(3)、链接断开,重新链接:
[html] view plain copy
print?
1. connectionStatus = MQTTConnectionStatus.NOTCONNECTED_UNKNOWN
REASON;
2. // inform the app that we are not connected any more, and are
3. // attempting to reconnect
4. broadcastServiceStatus(\"Connection lost - reconnecting...\");
(4)、由于其他的原因导致无法链接:
[html] view plain copy
print?
1. connectionStatus = MQTTConnectionStatus.NOTCONNECTED_UNKNOWNREASON;
2. // inform the app that we failed to connect so that it can update
3. // the UI accordingly
4. broadcastServiceStatus(\"Invalid connection parameters\");
(5)、链接成功:
[html] view plain copy
print?
1. broadcastServiceStatus(\"Connected\");// 状态---已连接
2. // we are connected
3. connectionStatus = MQTTConnectionStatus.CONNECTED;
(6)、未能链接上:
[html] view plain copy
print?
1. connectionStatus = MQTTConnectionStatus.NOTCONNECTED_UNKNOWNREASON;
2. // inform the app that we failed to connect so that it can update
3. // the UI accordingly
4. broadcastServiceStatus(\"Unable to connect\");
每一种链接的状态都会发送相应的状态信息,在log中打印出来,让我们很直观的看到此时的mqtt处于哪一种状态。
三、我们在链接mqtt的时候需要先定义一个链接的接口:
[html] view plain copy
print?
1. mqttClient = MqttClient.createMqttClient(mqttConnSpec,
2. usePersistence);
3.
String mqttConnSpec = \"tcp://\" + brokerHostName + \"@\"4. + brokerPortNumber;
详细信息是:“tcp://115.59.138.7@1883”;
[html] view plain copy
print?
1. private boolean connectToBroker()
上面的这个方法是判断当前的链接是否存在,存在返回true,不存在返回false,然后在执行后面的操作;
[html] view plain copy
print?
1. mqttClient.connect(generateClientId(), cleanStart, keepAliveSeconds);
此时才是mqtt建立链接的操作,第一个参数generateClientID()是建立链接的一个关键字,是为了区分不同的主机链接到mqtt,方便服务器那边进行区分。(“fiber1/Android手机自带的序列号,这个码是唯一的,每一部手机都是不一样的,这样才能更好的区分)。第二个参数是false。第三个参数是保持心跳,五秒一次。
上面就已经说过了,当它返回的是true的时候,就说明这个链接已经建立成功了,后续的操作就是订阅我们的主题了;
[html] view plain copy
print?
1. public void subOldDevices()// 重新订阅已绑定的设备
2. {
3. subscribeToTopics(mBindingDevices);
4. }
这个mBindingDevices是一个存放设备ID的数组;
[html] view plain copy
print?
1. private static ArrayList
mBindingDevices = new ArrayList (); 那么这个数组是什嚒时候传过来的呢?
[html] view plain copy
print?
1. public static void setConfigDevice(String deviceid)
2. {
3. mConfigDevice = deviceid;
4. }
5. public static ArrayList
getBindingDevices() 6. {
7. return mBindingDevices;
8. }
9. public static void setBindingDevices(ArrayList
bindingdevices) 10. {
11. mBindingDevices = bindingdevices;
12. }
13. public static void addBindingDevice(String deviceid)
14. {
15. if(deviceid !=null)
16. {
17. mBindingDevices.add(deviceid);
18. }
19. }
我们就需要查看之前的代码;这两个方法就是获取我们在之前的界面传过来的数据或者说程序启动后,从前后台那边获取的设备的数据;有了这个数组,那么就可以根据数组里面额数据取订阅主题了,我们所用到的QoS是1,目的是为了让我们接收到的消息是一定能够接收的到;
[html] view plain copy
print?
1. for (int i = 0; i < topicsString.size(); i++)
2. {
3. topics[i] = \"$TS/ALARM/\" + topicsString.get(i);
4. parms[i] = 1;
5. }
利用循环的方法将数据存到指定的数组中;
[html] view plain copy
print?
1. try
2. {
3. mqttClient.subscribe(topics, parms);// 订阅的主题嗯和QoS等级
4. subscribed = true;
5. }
上面的操作才是真正的订阅的操作;
既然有订阅,那么就必须有取消订阅,因为取消订阅是为了让我们的设备在注销之后,不想再收到任何的消息,或者是删除设备之后不想再收到任何的消息是一样的道理。所以这个操作是必不可少的。
[html] view plain copy
print?
1. public void unsubOldDevices()// 取消所有告警消息的订阅
2. {
3. unsubscribe(mBindingDevices);
4. }
取消订阅和订阅的区别就在于订阅的时候需要发送QoS,而取消订阅的时候是不需要发送QoS的;
[html] view plain copy
print?
1. try
2. {
3. mqttClient.unsubscribe(topics);
4. }
四、下面的就是接受广播:
1、
[html] view plain copy
print?
1. private class BackgroundDataChangeIntentReceiver extends BroadcastReceiver
我们在做这件事的时候,通过请求一个唤醒锁,我们要求尽可能的唤醒锁-只是足够维持处理器的运行,直到我们完成
2、
[html] view plain copy
print?
1. private class NetworkConnectionIntentReceiver extends BroadcastReceiver
所谓的响应于网络连接的变化-在连接到服务器之后,这让我们等待,直到我们有一个可用的数据连接
3、
[html] view plain copy
print?
1. public class PingSender extends BroadcastReceiver
这个广播用于唤醒手机,使手机可以时刻接收mqtt的推送消息;
4、
[html] view plain copy
print?
1. public class StatusUpdateReceiver extends BroadcastReceiver
当mqtt的客户端连接的状态发生改变的时候,通过StatusUpdateReceiver接收广播;
5、
[html] view plain copy
print?
1. public class MQTTMessageReceiver extends BroadcastReceiver
当收到mqtt的推送的消息时,通过MQTTMessageReceiver接收广播;
五、最后就是,当我们收到推送的消息,怎么处理?
通过异步任务,将获取的json数据进行解析,
[html] view plain copy
print?
1. public static class AlarmTask extends AsyncTask
2. {
3. private String jsonData = null;
4. private Context context = null;
5. int count = 0;
6. AlarmTask(String data, Context context)
7. {
8. this.jsonData = data;
9. this.context = context;
10. }
11. @Override
12. protected String doInBackground(Void... params)
13. {
14. ArrayList
curAlarmList = new ArrayList (); 15. mAlarmDataHelper = AlarmDataHelper.getInstance(context);
16. if (jsonData != null)
17. {
18. JSONArray jsonArray = null;
19. JSONObject jsonobj = null;
20. try
21. {
22. jsonobj = new JSONObject(jsonData);
23. }
24. catch (JSONException e1)
25. {
26. e1.printStackTrace();
27. return null;
28. }
29. String alarmData = null;
30. try
31. {
32. alarmData = jsonobj.getString(\"alarm\");
33. }
34. catch (JSONException e1)
35. {
36. e1.printStackTrace();
37. return null;
38. }
39. try
40. {
41. jsonArray = new JSONArray(alarmData);
42. }
43. catch (JSONException e)
44. {
45. e.printStackTrace();
46. return null;
47. }
48. if (jsonArray != null && jsonArray.length() > 0)
49. {
50. for (int i = 0; i < 1; i++)
51. {
52. JSONObject json = null;
53. long restore_time = 0;
. long alarm_time = 0;
55. try
56. {
57. json = (JSONObject) jsonArray.get(i);
58. }
59. catch (JSONException e)
60. {
61. e.printStackTrace();
62. }
63. try
. {
65. restore_time = json.getLong(\"restore_time\");
66. alarm_time = json.getLong(\"alarm_time\");
67. }
68. catch (JSONException e)
69. {
70. e.printStackTrace();
71. }
72. if (json != null)
73. {
74. if (restore_time > 0)
75. {
76. HistoryAlarm entity = AlarmParse
77. .parseHisAlarm(json);
78. mAlarmDataHelper.restoreAlarm(entity);
79. }
80. if (alarm_time > 0)
81. {
82. CurrentAlarm entity = AlarmParse
83. .parseCurAlarm(json);
84. Boolean result = mAlarmDataHelper
85. .checkCurrentAlarmID(entity
86. .getUnique_id());
87. CLog.e(\"------------curAlarmList already exist?: \
88. result);
. if (!result)
90. {
91. curAlarmList.add(entity);
92. count++;
93. MainApplication.addToastAlarm(
94. entity.getAlarm_id(), 1);
95. }
96. }
97. }
98. }
99. if (curAlarmList.size() > 0)
100. {
101. mAlarmDataHelper.insertCurrentAlarm(curAlarmList);
102. }
103. }
104. }
105. return null;
106. }
注意:在网络断开之后,我们没有重连的机制,只有在网络重新链接上之后,我们才会去再链接mqtt(重连的机制还是和第一次链接的机制一致),断网时,我们也没有必要去链接,因为这本身就是不通的,一直链接,反而会造成阻塞;
因篇幅问题不能全部显示,请点此查看更多更全内容
Copyright © 2019- huatuo3.com 版权所有 蜀ICP备2023022190号-1
违法及侵权请联系:TEL:199 1889 7713 E-MAIL:2724546146@qq.com
本站由北京市万商天勤律师事务所王兴未律师提供法律服务