引言

即便在通讯如此发达的今天,IM 也依然是诸多场景下非常重要的基础能力。因此做为 一名 Android 开发,不可避免的会遇到一些IM 相关的需求或问题。本文以一个Android开发的角度来讲述IM 开发相关的基础知识。

想要技术干货、行业洞察,欢迎关注网易云信博客。
了解网易云信,来自网易核心架构的通信与视频云服务。

IM开发需要面对的问题

  1. 网络问题,如何高效快速的传输数据?
  2. 协议问题,消息如何封装?
  3. 及时性问题,如何进行进程保活?

网络问题

TCP 的三次握手建立连接是一个非常耗时的过程。在 IM 场景下,数据的传输将会非常的频繁,如果每次传输都建立一个 TCP 连接,那么这个效率是不能接受的,并且频繁的建立连接可能会发生socket错误,所以我们需要 “复用”TCP连接,也就是平时所说的TCP长连接。
TCP 长连接
短连接在建立后,当数据传输完毕时会立即关闭,下次需要传输数据时需要重新建立连接,在日常的业务场景非常常见,比如通过 http/https 请求获取Server 数据。而长连接在传输完数据后并不会关闭,这样下次需要传输数据时就可以直接使用已经建立好的连接,这中间省去了连接建立的时间。但是建立一个 TCP 长连接却并不是“建立后不关闭”那么简单,因为 TCP 长连接会“被动”关闭。
网络地址转换 (NAT)
IPv4的容量是有限的,随着接入Internet的计算机数量的不断猛增,IP地址资源也就愈加显得捉襟见肘,于是也就产生了 NAT技术。简单来说,NAT就是在局域网内部网络中使用内部地址,而当内部节点要与外部网络进行通讯时,就在网关(可以理解为出口,打个比方就像院子的门一样)处,将内部地址替换成公用地址,从而在外部公网(Internet)上正常使用,NAT可以使多台计算机共享Internet连接,这一功能很好地解决了公共 IP地址紧缺的问题。通过这种方法,可以只申请一个合法IP地址,就把整个局域网中的计算机接入Internet中。
而我们就处于运营商(移动/联通/电信。。。)的局域网内。当我们接入运营商的网络后,会分配到一个运营商的内部 IP地址,于是我们就可以使用这个IP地址建立连接向外传输数据了。但是当这个IP闲置了一段时间(NAT超时时间)后,运营商为了节约资源,会把分配给我们IP回收掉。此时如果我们还继续使用之前那个未关闭的连接去传输数据,那么毫无疑问会失败的。下面是一些运营商的 NAT超时时间。
网络NAT超时时间中国移动3G/2G5 min中国联通2G5 min中国电信3G大于 28 min
要想长连接一直有效,那么闲置时间就不能太长,所以在闲置时我们需要向外(Server)传输一些数据包,这也就是常说的“心跳包”,用于告诉运营商这个 IP 还在被使用,告诉Server 客户端还在线。
心跳策略
心跳策略一般分为两种:
1. 固定心跳
2. 动态心跳
这里讲一下固定心跳,动态心跳可以参考 微信心跳 。固定心跳其实就是间隔固定时间发送一个心跳包。

心跳间隔 X 的值需要参考运营商的 NAT 超时时间确定,不能大于最小的 NAT超时时间,也不能太小,要不Server 的负担非常重。一般取一个比较接近最小的NAT超时时间,比如4分钟。

协议问题

协议决定是消息以什么样的形式传输,即发送时如果对消息进行封装,接收时如何解析。比如可以将消息体以 XML 的形式进行处理,这也就是 XMPP 协议,参考下面一个消息示意:
隔壁老王:你儿子长的比你帅多了。
老李:嘿嘿,谢谢夸奖!
<message><from>隔壁老王</from><to>老李</to><context>你儿子长的比你帅多了。</context><type>text</type>
</message><message><from>老李</from><to>隔壁老王</to><context>嘿嘿,谢谢夸奖!</context><type>text</type>
</message>复制代码
从上面的消息示意,我们可以发现一条消息的内容可以拆分成很多属性,而协议就是把这些属性组合起来。
以 XML 的形式传输消息,最大的一个问题,就是冗余数据太多了,特别是当消息的属性比较多时。
那么有没什么格式能尽可能有减小冗余数据?
其实无论消息如何封装,最终传输的肯定是二进制流,那么完全可以直接用二进制的形式对消息进行封装,这也就是二进制协议。下面是一个简单二进制协议的实现示意。
一条消息由from + to + context + type这几个属性组成,那么我们完全可以按顺序存储在二进制中,由于内容长度不确定,所以每个属性的开头我们可以使用固定字节数来记录这个属性的内容长度。当然,这里只是展示了一个二进制协议的例子,实际的消息会比这复杂多了,但是核心思路就是这么简单,最终无非是设计与实现形式上的差距。

及时性问题

IM的作为即时通讯,如果无法保证消息及时触达,那么意义就大打折扣。要保证消息及时触达,最关键要做到以下两点:
1. App 进程要尽量存活,也就是进程保活 ;
2. 在 App进程挂掉后,能够唤醒起来;
进程保活
进程保活其实是属于 Android 平台的一个话题,相信大家日常开发也遇到过,细节就不在这长篇大论了,简单的说下几个原则:
1. 优化内存,减小内存的占用,会大大的减小被 kill 的机率;
2. 多进程,将 UI 进程与 IM 进程独立出来,这样 IM 的进程负担就会小很多;
进程唤醒
严格意义上来说进程唤醒是属于进程保活的一个分支,这里单独列出来,是因为进程唤醒关注的是进程挂掉之后的动作。对于进程唤醒,这里也只列些原则,详细的可以去查阅相关资料。
1. 静态注册监听广播(<7.0)
2. Alarm定时任务,定时去检查进程是否存活。
3. JobScheduler定时任务,定时去检查进程是否存活( >5.0 )
4. 接入厂商推送
网易云信(NeteaseYunXin)是集网易18年IM以及音视频技术打造的PaaS服务产品,来自网易核心技术架构的通信与视频云服务,稳定易用且功能全面,致力于提供全球领先的技术能力和场景化解决方案。开发者通过集成客户端SDK和云端OPEN API,即可快速实现包含IM、音视频通话、直播、点播、互动白板、短信等功能。


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/news/276609.shtml
繁体地址,请注明出处:http://hk.pswp.cn/news/276609.shtml
英文地址,请注明出处:http://en.pswp.cn/news/276609.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

偷梁换柱做自己的封装系统

偷梁换柱做自己的封装系统&#xff01;菜鸟一开始都想把自己的信息加到系统里&#xff0c;但封装系统只会一点&#xff01;但我们可“拿来”&#xff0c;我们可以用偷梁换柱的方法来修改别人的系统&#xff0c;本文以雨林的GHOST5.0系统为例。一、准备工作1、当然是下载一个自己…

JQuery 1.6+ checkbox 状态选择

示例&#xff1a; HTML: <form><table><tr><td><input type"checkbox" id"select_all"/></td></tr><tr><td><input type"checkbox" name"select[]"/></td></…

台电u盘量产工具_简单几步,让U盘起死回生

如今&#xff0c;虽说云存储风靡&#xff0c;但U盘仍存在价值&#xff0c;毕竟在很多场合并不方便上网&#xff0c;即便如此网上存储有时也并不方便&#xff0c;也不安全。与此同时&#xff0c;如果是大文件存储&#xff0c;云盘上传和下载速度非常慢&#xff0c;并不适合海量数…

PXC集群常见错误(一)

欢迎关注MySQL 8.0必知必会系列课程。MySQL8.0必知必会-自动化部署 https://edu.51cto.com/course/16368.htmlMySQL8.0必知必会之参数标准化配置 https://edu.51cto.com/course/16358.html1.Cant start server: Bind on TCP/IP port: Address already in use…

获取GridView中RowCommand的当前选中行的索引或主键Id

获取GridView中RowCommand的当前索引行 前台添加一模版列,里面添加一个LinkButton前台 (如果在后台代码中用e.CommandArgument取值的话前台代码就必须在按钮中设置CommandArgument的值&#xff0c;值为绑定的数据库字段<asp:TemplateField HeaderText"操作"> …

系统架构师 项目经理 哪个更有前景_中央空调加地暖与五恒系统,哪个更省钱?...

每逢严冬酷暑,人们都会感叹空调是最伟大的发明,并且随着科技发展还在不断进化。从烤火取暖到空调和地暖的供暖,从纸扇电扇的吹风到空调的制冷,人们的需求正在不断提高,于是,为了满足人们的需求&#xff0c;市场上又衍生出了家装五恒系统。 恒温、恒湿、恒氧、恒洁、恒静这…

软件架构基本原则

软件架构本质上是绘制一幅复杂素描所打的草稿&#xff0c;我还说&#xff0c;如果你罩得住&#xff0c;可以不需要这个草稿。但这只是“理论上”&#xff0c;我们写软件&#xff0c;基本上不是在写只有几千行的代码的小程序&#xff0c;而是写数千万行的大型程序。《道德经》说…

Smarty目录结构和子目录路径问题

原文链接&#xff1a;http://bbs.csdn.net/topics/80223905 问题 1).最初在模板文件中使用了绝对路径&#xff08;相对于站点根的路径 ).如在system_info.tpl中图片是这样显示的<img src"/templates/default/images/logo.gif"/>这样当系统发布时&#xff0c;…

c++直角坐标系与极坐标系的转换_一篇阅读量高达2百6十多万的关于坐标系和投影的相关知识探讨...

本文转载于CSDN作者rsyaoxin这是一篇关于坐标和投影的「神文」截止目前浏览量已达2698239是相关文章中不可打破的神话...文末有本文作者推荐的两款坐标转换的小工具下载链接回想一下&#xff0c;接触遥感专业也有几个年头了&#xff0c;而现在越来越偏离遥感了&#xff0c;突然…

c# xmlhttp POST提取远程webservice数据

public string cancel(string StudentID,string Mobile){string datanull;data "<?xml version1.0 encodingutf-8?><soap:Envelope xmlns:xsihttp://www.w3.org/2001/XMLSchema-instance xmlns:xsdhttp://www.w3.org/2001/XMLSchema xmlns:soaphttp://schemas.…

Javascript实现的左右滑动菜单

原文链接&#xff1a;http://www.cnblogs.com/miqi2214/archive/2009/04/28/1445308.html Javascript实现的左右滑动菜单 先看Demo便于理解&#xff1a;请大家往这里看 业务需求&#xff1a; 1. 菜单个数不固定&#xff0c;当菜单个数长度小于给定范围宽度&#xff0c;则左右移…

查询用户分配角色TCODE

s_bce_68001419 转载于:https://www.cnblogs.com/yangliang/archive/2009/04/13/1434862.html

query string parameters什么意思_public static void main(String[] args) 是什么意思?(转)...

public static void main(String[] args)&#xff0c;是java程序的入口地址&#xff0c;java虚拟机运行程序的时候首先找的就是main方法。一、这里要对main函数讲解一下&#xff0c;参数String[] args是一个字符串数组&#xff0c;接收来自程度序执行时传进来的参数。如果是在控…

经典算法:位图排序

最近发现一个有趣的排序算法&#xff0c;通过位图来完成排序。位图排序其实就是基数排序&#xff0c;只不过位图排序的下标是比特位。 问题描述 输入&#xff1a;一个最多包含n个正整数的文件&#xff0c;每个数都小于n&#xff0c;其中n10^7。如果在输入文件中有任何正数重复出…

PHP中删除目录的三种方法

原文链接&#xff1a;http://www.chinaz.com/program/2008/1022/41645.shtml PHP中删除目录的三种方法 1、递规法&#xff1a;利用递归一层一层的删。 deleteDir(&#xff04;dir) { if (rmdir(&#xff04;dir)false && is_dir(&#xff04;dir)) {if (&#xff04;d…

b样条曲面绘制 opengl_CAD制图软件中如何利用EXCEL输入坐标绘制曲线?

当在使用浩辰CAD制图软件绘制图纸的过程中&#xff0c;经常要绘制由多个坐标点连接成的曲线时&#xff0c;有什么方便快捷的方法吗&#xff1f;那当然是有的。利用EXCEL表格保存数据并与CAD制图软件巧妙地结合起来&#xff0c;就能很容易地画出曲线。下面给大家详细介绍一下吧&…

根据进程名杀掉进程

foreach (System.Diagnostics.Process pro in System.Diagnostics.Process.GetProcesses()){if (pro.ProcessName "Bss"){pro.Kill();break;}} 转载于:https://www.cnblogs.com/wolfcool/archive/2009/04/17/1438284.html

JavaScript 操作 Cookie

从事web开发也有些日子了&#xff0c;cookie 是个啥差不多能说明白&#xff0c;可是实际自己一上手操作就是得去搜索(你们懂的)&#xff0c;结果被鄙视了...所以就写一篇博文做为自己的学习笔记&#xff0c;嘿嘿&#xff0c;博客的好处在此体现出来了。 什么是 Cookie “cookie…

阿里云服务器购买该如何选择?阿里云服务器购买步骤流程介绍...

很多第一次购买阿里云服务器&#xff0c;不知该如何选择适合自已的服务器。其实购买阿里云服务器&#xff0c;主要是根据自已网站的流量来决定的。如果网站流量不大&#xff0c;一天只有几百ip&#xff0c;一般选择1核cpu&#xff0c;1G内存&#xff0c;1MB带宽就可以用了&…

python 切片_全面解读Python高级特性切片

大家好&#xff0c;欢迎来到Crossin的编程教室&#xff01;众所周知&#xff0c;我们可以通过索引值(或称下标)来查找序列类型(如字符串、列表、元组…)中的单个元素&#xff0c;那么&#xff0c;如果要获取一个索引区间的元素该怎么办呢&#xff1f;切片(slice)就是一种截取索…