用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

  1. 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里包含了clientjar包,所以就不用去sync下载了。

      

推荐使用:

eclipse出品Android Demo

https://github.com/eclipse/paho.mqtt.android

 

 

  1. 2初始化

mqttClient的生命周期不应该绑在某个activity上,建议传getApplicationContext(),使其与整个应用共存亡(本质是一个service),应当持续在后台监听消息的收发)

 

初始化一个mqttClient,实际上整个应用也只需要存在一个mqttClient

connectiOption默认构造器属性,userNamepassword找后台要。

自动重连是默认关闭的,设置开启后,会在掉线的情况下每隔1秒请求一次;

设置超时,setConnectionTimeouttimeout默认是30秒;

cleanSession,默认开启,每断一次,就清除这个链接,方便后台管理。

经调试,建议关闭自动重连,在mqttCallbackExtenedMqttActionListener里做重连处理。

  1. 开始连接

client设置mqttCallbackExtened回调和MqttActionListener监听

mqttCallbackExtened需要实现的方法。这里的messageArrived貌似并未响应接收的消息队列。

重点关心以下

connectComplete

如果需要保持某个会话长时间保存,那么在connectOptionsetCleanSessionfalse,

因为业务要求,我这里并不需要保存长会话,断线后就重新生成id,重新订阅,服务器也不用管理断线的会话,定时清理,减少后台压力。

  connectionLost

   之前有提到,不设置自动重练,那么应该要做手动重连的处理。
  实际上这篇总结最重要的也就是这里,什么时候重连?重连做什么?

   我这里先保存了断开连接的日志,方便查错,然后setClientID----重新生    成id用于建立新会话,5秒后请求连接。

mqttActionListener需要实现的方法。

同样的,在成功后订阅主题。

失败,记录日志,5秒后重连。
重新订阅主题----后文会说这里的订阅方案。

问题来了:

为什么不在mqttCallBackExtendonSuccess里订阅主题,而要在Listener里订阅?

mqttCallBackExtendonSuccess先被触发,

然后才走IMqttActionListeneronSuccessonFailure

可以认为,只有后者成功走onSuccess,整个订阅才算成功。

或者说,mqttCallBackExtend着眼在client和服务器的连接是否成功,不管会话的建立是否成功,后者重点在会话的建立。(这里有点模糊,我有空再查查资料)

 

  1. 4订阅主题

 其实这里订阅只需要设置IMqttMessageListener就可以了,

messageArrived方法处理接收到的消息,根据接收到的topic进行分类处理。

此处我是用EventBus分发所有接到的消息,在fragment或者Activity里处理对应topic

 (EventBus轮子的拆解会有的)

mqtt的消息类型(messageId用来保证该条消息的唯一性,可做去重判断)


    订阅重点:
    一:在订阅的时候,需要注意的是,重复订阅相同主题没有问题,且相比重新连接消耗低。
在保证不断线的情况下:


A循环操作  订阅,取消订阅,订阅,取消订阅

和循环操作开链接,订阅,取消订阅,关连接。

     前者消耗比后者小很多。

     二:同一个用户可以同时订阅多个主题,不同的用户订阅的主题也不一样,对消息要分类处理。

这里采取的方案是,每订阅一个主题topic,就把这个topic存入一个list

取消订阅则把该topiclist中移除,

遇到断线需要重新连接的时候,循环订阅该list中的主题。

    当然,mqqttClient.subscribe()提供了一次订阅多个主题的方法,当然也需要同时传入多个IMqttMessageListener,这相当于要开发者同时维护几个list(当然也可以写个map

    这样我就只需要关心该名用户是否订阅这个消息,消息由eventBus发送出去,在相应的类里处理。
    打个比方:甲乙丙三人分别订阅了ABC三种报纸,
    当报纸message送到传达室client的时候,

    前者是:传达室大爷client喊一句报纸来了,甲乙丙class通过报纸的名字topic来拿自己那份报纸message,不是自己的topic就不处理。
    

    后者是:传达室大爷client根据 当初订阅的表格topicList,找到对应的送报员messageListener,分发ABC报纸,甲乙丙class拿各自的message处理;

     
    我这里选择前者,传达室大爷不用找各自的送报员,只管拿到订阅表里的报纸,报纸都送过去,是你的就处理,不是你的就丢弃,是个豪爽的传达室大爷。
    

  1. 5取消订阅
    比较简单。

 

  1. 6释放资源

从搭建mqtt的准备工作就可以看出,我们用的eclipse这个demo,本质上就是一个service,如果绑定的是某ActivityActivity被干掉,当然就has leaked了。(Service被干调,但连接一直保持)

顺序调用client.close –client.unRegisterResource

 

敲黑板:可我干嘛要释放这个资源呢?我初始化的时候,绑定的就是整个应用的生命周期,service一直在后台运行着,接收订阅的消息,

开发者只要关心:用户订阅了哪个主题,又取消订阅了哪个主题,在哪个界面处理哪个主题。


==============================================

以上就是使用eclipse出品的mqttAndroidClient通过订阅主题,实现mq消息**,实现IM的方法。


0 0