基于UDP实现多媒体即时通信机制

最近有个项目需要仿照QQ设计一个高性能即时通信应用,其中涉及到传输图片、视频、大块文字的需求。为了实现高性能的同时减少带宽占用,我们参考了QQ通信协议以及Google Protocol Buffers的精简设计理念设计了一套基于UDP的多媒体即时通信机制,达到低带宽、高性能的效果。

QQ通信协议

QQ在即时通信应用中并没有采用XMPP、HTTP等上层的协议进行消息通信,主要是基于效率的考虑:XMPP、HTTP等上层的传输协议为了保证易用性、通用性,包含了太多对应用无用的数据,从带宽、性能方面而言并不是特别优秀。QQ通信的数据结构和Protocol Buffers的机制类型,通过对数据进行序列化反序列化直接进行传输。但是QQ的通信仍然是基于TCP的,而我们项目的场景要求对流量使用有着变态的要求,因此我们尝试基于UDP的方式底层机制来实现。

目标与问题

项目要求传输的数据至少包括如下数据类型:文字、语音、图片、视频、文档、其他等。

文字 语音 图片 视频片段 文档 表情 其他
由于UDP协议中每个包传输的数据大小有限并且UDP是不可靠的通信传输方式,会导致出现粘包和丢包的情况,设计不当的话其效率将比TCP/HTTP这类传统传输协议低效得多,因此在实现中既要处理好粘包的情况,也要处理好丢包的情况。

数据结构设计

在协议字段上,我们参考了QQ的TCP传输协议,将数据进行分片打包的方式进行流式处理。

序号 字段 类型 描述
1 tag byte[2] 传输协议包头固定值0xbebe,用于区分是否有效数据包、过滤部分恶意攻击
2 counter int 数据包唯一ID,单调递增以帮助接收端识别丢包
3 cmdType byte 消息类型,不同的消息ID对应不同的数据内容(如文字、语音、图片、视频、文档等等)
4 sliceID short int 分片序号,第一片数据序号要加上0xff用于区别首包数据,分片序号采用降序递减的方式发送(避免接收方区分EOF);不分片传输时分片序号为0xffff (节省一个byte)
5 sliceLength short int 待传输数据的整体长度或分片数据长度,取决于是否是首片数据。首片数据时传整体长度是为了接收方预准备缓存以避免多次数据拷贝操作
6 body byte[] 消息体具体内容。分片传输时首片数据的body提供配置信息,依次为:MTU值、丢包处理、超时时间,对应字段为:short int, byte, short int
代码如下:

public abstract class AbstractPacket {
/** 协议包头固定值0xbebe,用于区分是否有效数据包,防止恶意攻击 /
protected byte[] tag;
/
数据包唯一ID,不断递增,帮助识别丢包 /
protected int counter;
/
消息类型,不同的消息ID对应不同的数据内容(如文字、语音、图片、视频、文档等)/
protected byte cmdType; /消息分片序号,第一片数据的序号要加上0xff用于区别首包数据,序号采用降序递减的方式发送(避免区分EOF) /
protected short int sliceID; /
待传输数据的整体长度或分片数据的长度 /
protected short int sliceLength;
/
消息体*/
protected byte[] body;

}
以上数据结构是经过验证的较优结果,能够以有限的字段传递足够的信息(数据本身、丢包、超时、配置信息等)。

数据传输

数据传输有两种情形:

1、普通数据传输

要发送的数据较小时,不需要分片就可直接发送,这是最简单的一种情形。例如要传输一串文本(”QQ”)时,那么数据包的设置将为:tag=0xbebe, counter=<递增ID>,cmdType=<文字消息ID>,sliceID=0xffff, sliceLength=2, body=”QQ’。

2、大数据传输

如果要发送的数据很大,一次UDP发送无法发送完毕,那就需要对数据进行分片。例如要发送的一段大小为10KB的语音数据,那么

首片数据的设置将为:tag=0xbebe, counter=<递增ID>,cmdType=<语音消息ID>,sliceID=0xff+10KB/1024+1=0x10A, sliceLength=5, body=”MTU=1024,Relay=1,Timeout=120″,

第二片数据的设置为:tag=0xbebe, counter=<递增ID+1>,cmdType=<语音消息ID>,sliceID=10KB/1024=0xA, sliceLength=1024, body=”<语音消息前1024个字节>”,

最后一片数据的设置为:tag=0xbebe, counter=<递增ID+11>,cmdType=<语音消息ID>,sliceID=1, sliceLength=1024, body=”<语音消息最后1024个字节>”

丢包处理

由于UDP传输存在不可靠、乱序等问题,数据传输的接收方需要有效的识别丢包并反馈给发送方以请求重传。丢包处理有两种情况:第一种情况,针对不需要分片的数据,只要过了超时时间即可要求发送方重传。

第二种情况是针对分片的大数据,一种最简单的策略(Relay=1)是一旦发现分片数据丢失即要求所有分片数据重新传输,这种方式对接收方而言最简单,但是效率很低。实测发现其效率远低于使用TCP等传统的传输方式。另一种优化的策略(Relay=2)是只要求发送方重传丢失的分片数据,已收到的分片数据不影响,这种策略也是大多数丢包处理方案所采用的的方式。两种方式进行结合也是可以的,主要看分片数量的大小:如果分片数量很少,第一种策略效果没问题;如果分片数量很大,则应采用第二种。

除了以上两种情况,丢包处理还可以进一步进行优化,具体的实现方法可以参考TCP的sliding window的重传方式,但由于实现相对复杂,以后再行实现。

命令回复

数据传输的回复结果理论上可以复用数据发送的流程,但由于命令的回复内容都较短,一般不会出现需要分片的情景,只需要处理丢包的情景,因此可以针对数据发送的流程进行优化来实现命令回复。具体如下所示:

序号 字段 类型 描述
1 tag byte[2] 传输协议包头固定值0xbeef,用于区分是否有效数据包、过滤部分恶意攻击
2 counter int 回复数据包唯一ID,单调递增以帮助识别丢包
3 cmdType byte 消息类型,不同的消息ID对应不同的数据内容(如文字、语音、图片、视频、文档等等)
4 result short int 信息传输的返回码,具体可根据需要随便定义
5 body byte[] 信息传输的成功反馈或错误解释
确认MTU的值

选好MTU的值对于传输效率的影响十分巨大,因此在协商过程中应该把MTU的值设为多少很有讲究。MTU选得过大,则丢包率严重;MTU选得过小,则传输次数太多,影响传输速度(因为重传机制存在)。在我们大量的测试验证中发现,MTU在500字节左右是最合适的。如果网络环境较差,丢包率较高,可以适当减少MTU的值,但不要低于200;如果网络环境较好、丢包率低,可以适当增大MTU,但不要大于1500。

方案效果

虽然我们花了一些时间另辟蹊径地在UDP的基础上设计了这个大数据传输方案,期望得到相对更优的方案。上线后发现整体优化结果不是非常明显,甚至在一些情况下效率反而很低,可能是丢包重传机制没有优化的原因。重新发明轮子果真是一件吃力不讨好的事情。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,864评论 6 494
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,175评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,401评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,170评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,276评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,364评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,401评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,179评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,604评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,902评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,070评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,751评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,380评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,077评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,312评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,924评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,957评论 2 351