1. 树叶网络科技
  2. 新闻圈子
  3. 【融云分析】IM即时通讯之链路保活

【融云分析】IM即时通讯之链路保活

2019-04-04 09:08 / 树叶网络科技/ 技术分享

众所周知,IM即时通讯是一项对即时性要求非常高的技术,而保障消息即时到达的首要条件就是链路存活。那么在复杂的网络环境和国内安卓手机被深度定制化的条件下,如何保障链路存活呢?本文详解了融云安卓端SDK在基于TCP协议实现链路保活方面的探索和经验。

IM系统整体框架

如上图所示,为了保障链路存活,一套成熟的IM系统一般会包含消息链路和推送链路两条长连接通道。当有新消息到达时,消息服务首先会判断消息链路是否存活,假如消息链路处于存活状态,消息优先从消息链路下发到客户端,否则会被路由到推送服务器,由推送链路下发。

综上所述,链路保活涉及到消息链路和推送链路两条链路的保活策略。基于这两条链路使用场景的不同,保活策略上除了心跳机制是相同的,其它保活策略各有不同。下面将详细讲解。

链路保活的必要性

基于TCP的Socket连接建立之后,假如不做任何处理,这个连接会长时间存在并且可用吗?答案是否定的。原因有两点:

一、默认Socket连接无法及时探测到链路的异常情况,即使将Socket的属性参数keepAlive设置为true仍然无法及时获取到链路存活状态。这是因为Socket的连接状态是由一个状态机进行维护的,连接完毕后,双方都会处于建立状态。假如某台服务器因为某些原因导致负载超高,无法及时响应业务请求,这时TCP探测到的仍然是连接状态,而实际上此链路已经不可用了。

二、国内运营商的NAT超时机制会把一定时间内没有数据交互的连接断开,这个时间可能只有几分钟,远无法满足我们的长连接需求。

通用保活机制-心跳机制

基于以上原因,要维持Socket连接长时间存活,就需要实现自己的保活机制。最通用的一种保活机制就是心跳机制。即客户端每隔一段时间给服务器发送一个很小的数据包,根据能否收到服务器的响应来判断链路的可用性。为了节省流量,这个包一般非常小,甚至没有内容。

那么客户端如何实现定时发送心跳包呢?一般有两种方式:

一种是通过Java里的Timer来实现。最基本的步骤如下:

1、建立一个要执行的任务TimerTask。

2、创建一个Timer实例,通过Timer提供的schedule()方法,将TimerTask加入到定时器Timer中,设置每隔一段时间执行TimerTask,在TimerTask里发送心跳包。这种方式实现起来较简单,而且省电,不需要持有WakeLock。缺点也很明显,长时间在后台,进程被回收或者系统休眠后,Timer机制随之失效。

另外一种方式是利用安卓系统的定时任务管理器AlarmManager循环执行发送心跳包的任务。这种方式不会因为系统休眠而失效,系统休眠后仍然可以通过WakeLock唤醒,执行心跳任务,因此相对Timer机制,这种方式比较费电,使用的时候一定要注意如下几点:

首先根据需求合理使用AlarmManager的闹钟参数。闹钟各参数的区别参考下表:

其次AlarmManager提供了cancel()方法,在设置新的定时任务前,通过cancel()方法取消系统里设置的同类型任务,避免设置冗余任务。

最后,安卓从6.0版本引入了Doze模式,并提供了新的闹钟设置方法setExactAndAllowWhileIdle(),通过该方法设置的闹钟时间,系统会智能调度,将各个应用设置的事务统一在一次唤醒中处理,以达到省电的目的。推荐在安卓6.0以上系统中,优先使用该方法。

消息链路保活机制

消息链路作为收发消息的主要通道,需要最大程度保障链路的可用性。在链路不可用或者异常断开时,能及时探测并启动重连等保障机制。基于以上特性,消息链路除了前面所说的心跳机制外,还另外维护了两套链路优化机制:复合连接机制和重连机制。

复合连接机制的基本步骤如下:

1.客户端连接导航服务器,导航服务器会下发应用对应的配置信息,其中包括连接服务器的地址列表。

2.客户端从第一个服务器地址尝试连接,并启动超时机制,假如连接失败或没有及时收到服务响应,则继续尝试连接下一个直到成功连接,将成功连接的地址保存到本地,作为最优地址,后面连接时优先使用此地址。通过这种机制,能保障客户端优先选用最优链路,缩短连接时间。

重连机制,则是指业务层在检测到与服务器的连接断开后,尝试N次重新连接服务器,首次断开1秒后会重新连接,假如仍然连接不成功,会在2秒后(重连间隔时间为上次重连间隔时间乘2)尝试重新连接服务器,以此类推当尝试重连N次后,仍然连不上服务器将不再尝试重新连接,只有在网络情况发生变化或重新打开应用时才会再次尝试重连。

推送链路保活机制

推送链路作为消息到达的补充手段,要求尽可能延长在后台的存活时间。即使被杀后,仍然能被再次唤醒。iOS手机有APNS来达到以上效果,但安卓的官方推送系统FCM在国内基本不可用。那在国内安卓系统上如何保障推送到达呢?首先咱们需要先了解下安卓系统上进程管理的两大机制:

一种是LMK机制,英文是LowMemoryKiller,基于Linux的内存管理机制衍生而来。主要是通过进程的oom_adj值来判定进程的重要程度,从而决定是否回收这些进程。oom_adj的值越低,代表重要度越高,比如native进程,framework层启动的系统进程,优先级一般都为负数。其次是前台可见进程,系统也不会回收。然而可见进程退到后台后,oom_adj的值会立即升高,在系统定时清理时被杀。

另外一种机制是安卓原生的权限管理机制(AppOps),各大厂家在此基础上又进行了深度定制化,比如小米的安全中心,华为的手机管家等,都用来进行权限管理。该权限管理机制运行在安卓系统的框架层,上层各应用的进程假如想尝试重新启动,系统首先会去权限管理中心检查该进程有没有自启动权限,假如有,才准予启动。否则,从框架层直接限制系统的启动。

基于以上两种机制,推送链路的保活也可分为两大类,

一进程保活。它的思路是根据LMK机制提高进程优先级,降低被杀的几率。主要有以下几种方法:

监听黑屏事件,启动1像素透明Activity,使应用进程转为可视进程,降低被杀概率。在屏幕亮时,关闭该Activity。

双服务守护。A服务以startForeground()形式启动,发送一个通知,B服务同样以startForeground()形式启动,且发送和A相同ID的通知,然后在B服务里调用stopForeground()方法,取消通知。这样A服务就会以前台进程的形式存活,且不影响用户感知。

根据文件锁互斥原理,监视Java进程存活状态,若被杀,Linux层成功持有文件,则通过exec()命令,打开一个纯Linux的可执行文件,开启一个Daemon进程,该进程因为从Linux层启动,在安卓5.0之前,优先级会比较高,不会被杀。在安卓5.0之后,该方式不再有效。

二进程拉活的策略和安卓系统的AppOps机制有关,一般有如下几种:

一、利用Service本身的Sticky属性,在Service的onStartCommand()中返回START_STICKY,这样当Service被杀掉后,系统会自动尝试重启。不过在国内定制化的系统上,这种方式能成功重启的几率很低,需要用户在权限管理中心打开自启动等权限,才能成功拉活。

二、也就是前面讲过的心跳机制,不过这里要求使用AlarmManager设置ELAPSED_REALTIME_WAKEUP属性的闹钟,在系统休眠后,才会正常接受到心跳事件,从而将进程拉活。

三、通过监听网络切换,用户行为等事件,拉起进程。

四、应用间互相拉活。比如系统里有好几个应用集成了同一个SDK,那么在用户启动其中某一个App的时候,SDK会去扫描其它应用,把"兄弟姐妹"拉活。这种方式对用户使用感受伤害非常大,会造成系统莫名其妙的耗电。

随着安卓系统版本的迭代,对后台进程的启动管控越来越严。为了解决推送的问题,各手机厂家推出了自己的系统级推送服务。由厂家在Framework层统一维护一条推送通道,上层所有应用共同使用该推送链路,不需要再维护单独进程。当前支持系统级推送的厂家有:小米、华为、魅族、vivo、OPPO,这种系统级别的推送省电,省内存,到达率高。应用可以根据手机型号的不同,优先使用厂家系统级别的推送,再配合自身的保活机制,最大程度保障推送的到达率。

集成第三方系统级推送之后,整个消息的收发流程可以参考下图:

扫一扫关注树叶网络科技公众号

扫一扫关注树叶网络科技公众号

相关文章

【树叶科技与众不同】凭借对设计的热爱和执着,互联网营销趋势的敏锐洞察和深刻理解,与众多同行不同的是,树叶科技更注重与客户互促共生,价值同在。
本圈子所有内容若需转载请 联系我们。

客服中心

(点击按钮可通过QQ进行沟通,请确认启动QQ。)

关闭
业务相关,请咨询售前客服。为避免丢失消息,请尽量添加好友
周经理:
13559621427

处理:投诉、建议、代理、大客户

我们会全力以赴满足您的服务请求

公司地址:泉州市晋江青阳街道顶立大厦3楼
电话:13559621427

Hi,您需要什么服务?我来帮您!

Hi,Are you ready?

准备好开始了吗?
那就与我们取得联系吧

有一个互联网项目想和我们谈谈吗?您可以填写右边的表格,让我们了解您的项目需求,这是一个良好的开始,我们将会尽快与你取得联系。当然也欢迎您给我们写信或是打电话,让我们听到你的声音!

SHUYE NET 树叶网络互联网整合解决方案

地址:泉州市晋江青阳街道顶立大厦3楼

业务热线:13559621427

大客户专线:13559621427

售前QQ:有问题请联系我 有问题请联系我 有问题请联系我 有问题请联系我

E-mail:Admin@0595zx.net

合作意向表

您希望我们为您提供什么服务?

预算