用MqttAndroidClient实现IM
来源:互联网 发布:淘宝卖家在哪里看等级 编辑:程序博客网 时间:2024/06/06 16:34
转载自:https://sanwen8.cn/p/6e2mvAG.html
用MqttAndroidClient实现IM
(只讲实现,不讲协议)
0. MQTT是什么?
mqtt的官方文档
http://mqtt.org/documentation
Github上有中文翻译
https://github.com/mcxiaoke/mqtt
以上文档均为mqtt协议的说明,没有具体如何使用的指导,深入研究协议的可以看。
1.Why MQTT?
某云的文档是这样说的
(原文链接
https://help.aliyun.com/document_detail/42419.html )
总结起来:方便,相比socket更加轻便。如果不想用第三方的库(比如环信融云等老牌大厂),那就让后台搭个服务器,自己做IM,少很多限制。
2. 集成 MqttAndroidClient
2.1准备工作
实际上在gradle sync的时候,service包会报异常
『Error:Failed to resolve: org.eclipse.paho:org.eclipse.paho.android.service:1.1.0』
所以建议是在github上下载Android Demo,把整个service 包当一个单独的module,作为jar包,编译到你的项目中
service里包含了client的jar包,所以就不用去sync下载了。
推荐使用:
eclipse出品Android Demo
https://github.com/eclipse/paho.mqtt.android
2初始化
mqttClient的生命周期不应该绑在某个activity上,建议传getApplicationContext(),使其与整个应用共存亡(本质是一个service),应当持续在后台监听消息的收发)
初始化一个mqttClient,实际上整个应用也只需要存在一个mqttClient。
connectiOption默认构造器属性,userName和password找后台要。
自动重连是默认关闭的,设置开启后,会在掉线的情况下每隔1秒请求一次;
设置超时,setConnectionTimeout,timeout默认是30秒;
cleanSession,默认开启,每断一次,就清除这个链接,方便后台管理。
经调试,建议关闭自动重连,在mqttCallbackExtened和MqttActionListener里做重连处理。
3 开始连接
client设置mqttCallbackExtened回调和MqttActionListener监听
mqttCallbackExtened需要实现的方法。这里的messageArrived貌似并未响应接收的消息队列。
重点关心以下
connectComplete
如果需要保持某个会话长时间保存,那么在connectOption里setCleanSession为false,
因为业务要求,我这里并不需要保存长会话,断线后就重新生成id,重新订阅,服务器也不用管理断线的会话,定时清理,减少后台压力。
connectionLost
之前有提到,不设置自动重练,那么应该要做手动重连的处理。
实际上这篇总结最重要的也就是这里,什么时候重连?重连做什么?
我这里先保存了断开连接的日志,方便查错,然后setClientID----重新生 成id用于建立新会话,5秒后请求连接。
mqttActionListener需要实现的方法。
同样的,在成功后订阅主题。
失败,记录日志,5秒后重连。
重新订阅主题----后文会说这里的订阅方案。
问题来了:
为什么不在mqttCallBackExtend的onSuccess里订阅主题,而要在Listener里订阅?
mqttCallBackExtend的onSuccess先被触发,
然后才走IMqttActionListener的onSuccess或onFailure
可以认为,只有后者成功走onSuccess,整个订阅才算成功。
或者说,mqttCallBackExtend着眼在client和服务器的连接是否成功,不管会话的建立是否成功,后者重点在会话的建立。(这里有点模糊,我有空再查查资料)
4订阅主题
其实这里订阅只需要设置IMqttMessageListener就可以了,
messageArrived方法处理接收到的消息,根据接收到的topic进行分类处理。
此处我是用EventBus分发所有接到的消息,在fragment或者Activity里处理对应topic
(EventBus轮子的拆解会有的)
附mqtt的消息类型(messageId用来保证该条消息的唯一性,可做去重判断)
订阅重点:
一:在订阅的时候,需要注意的是,重复订阅相同主题没有问题,且相比重新连接消耗低。在保证不断线的情况下:
A循环操作 订阅,取消订阅,订阅,取消订阅…
和循环操作开链接,订阅,取消订阅,关连接。
前者消耗比后者小很多。
二:同一个用户可以同时订阅多个主题,不同的用户订阅的主题也不一样,对消息要分类处理。
这里采取的方案是,每订阅一个主题topic,就把这个topic存入一个list,
取消订阅则把该topic从list中移除,
遇到断线需要重新连接的时候,循环订阅该list中的主题。
当然,mqqttClient.subscribe()提供了一次订阅多个主题的方法,当然也需要同时传入多个IMqttMessageListener,这相当于要开发者同时维护几个list(当然也可以写个map)
这样我就只需要关心该名用户是否订阅这个消息,消息由eventBus发送出去,在相应的类里处理。
打个比方:甲乙丙三人分别订阅了ABC三种报纸,
当报纸message送到传达室client的时候,
前者是:传达室大爷client喊一句报纸来了,甲乙丙class通过报纸的名字topic来拿自己那份报纸message,不是自己的topic就不处理。
后者是:传达室大爷client根据 当初订阅的表格topicList,找到对应的送报员messageListener,分发ABC报纸,甲乙丙class拿各自的message处理;
我这里选择前者,传达室大爷不用找各自的送报员,只管拿到订阅表里的报纸,报纸都送过去,是你的就处理,不是你的就丢弃,是个豪爽的传达室大爷。
5取消订阅
比较简单。
6释放资源
从搭建mqtt的准备工作就可以看出,我们用的eclipse这个demo,本质上就是一个service,如果绑定的是某Activity,Activity被干掉,当然就has leaked了。(Service被干调,但连接一直保持)
顺序调用client.close –》client.unRegisterResource
敲黑板:可我干嘛要释放这个资源呢?我初始化的时候,绑定的就是整个应用的生命周期,service一直在后台运行着,接收订阅的消息,
开发者只要关心:用户订阅了哪个主题,又取消订阅了哪个主题,在哪个界面处理哪个主题。
==============================================
以上就是使用eclipse出品的mqttAndroidClient通过订阅主题,实现mq消息**,实现IM的方法。
- 用MqttAndroidClient实现IM
- mqttandroidclient connect 调用序列:
- IM即时通讯实现原理
- IM即时通讯实现原理
- IM服务器架构实现
- IM服务器架构实现
- IM即时通讯实现原理
- IM即时通讯实现原理
- XMPP实现IM
- IM即时通讯实现原理
- IM即时通讯实现原理
- openfire+android实现IM
- IM即时通讯实现原理
- IM即时通讯实现原理
- IM服务器架构实现
- IM即时通讯实现原理
- IM即时通讯实现原理
- IM即时通讯实现原理
- Shell脚本接收有空格参数问题
- c# 线程中如何控制控件焦点
- Google glog
- Javascript操作DOM常用API总结
- svn
- 用MqttAndroidClient实现IM
- mysql5.7配置文件优化
- 数学工具笔记之一:一张图告诉你什么是时域和频域
- informix 分片表
- Android ScrollView 和ListView 一起使用的问题汇总
- 反射
- 万能的GridView,定制你想要的网格线
- PHP获取HTTP头信息
- 重新blas库函数