iOS自建IM相关

一、涉及到的第三方库

1、GCDAsyncSocket

GCDAsyncSocket是一个封装好的,帮助开发者完成socket的通信过程。数据上传以及接收。

创建scoket对象后,遵循它的代理,里面有一个最重要的方法: 接受解析服务器数据

- (void)socket:(GCDAsyncSocket *)socket didReadData:(NSData *)data withTag:(long)tag {
    if (tag == xxx1) {
        <!--DataInputStream是一个从二进制data取数据的类,以约定的格式-->
        DataInputStream *input = [[DataInputStream alloc] initWithData:data];
        TCPHeaderFrame *header = [[TCPHeaderFrame alloc] init];
        <!--约定的格式:1-2字节为header内容长度-->
        header.headLength = [input readShort];
        <!--约定格式:3-4字节为版本号-->
        header.version = [input readShort];
        <!--约定格式:5-6字节为aid-->
        header.cid = [input readShort];
        <!--约定格式:7-10字节为bid-->
        header.bid = [input readInt];
        <!--约定格式:11-12字节为body长度-->
        header.bodyLength = [input readShort];
        <!--约定格式:以上定义了12字节的header内容 剩下的就是具体的消息内容了-->
        if (header.bodyLength <= 0) {
            [socket disconnect];
        } else {
            <!--这里从header中获取到消息内容长度 开始读取 调用该方法会再次进入此代理方法 tag:xxx2-->
            [socket readDataToLength:header.bodyLength withTimeout:-1 tag:xxx2];
        }
    } else if (tab == xxx2) {
        [[LocalDataReciever sharedInstance] handleProtocal:data cid:self.cid];
    } else {
        [socket disconnect];
    }
}
<!--这里根据cid判断是我们定义的哪种消息 或者说是哪种命令-->
- (void) handleProtocal:(NSData *)body cid:(int16_t)cid {
    if (cid == xxx) {
        <!--假如这里是Person消息内容 那么就用Person类解析-->
        Person *p = [Person parseFromData:data error:nil];
        //执行相关逻辑
    }
}

接下来是如何使用scoket上传数据:下面是一个要上传给服务器的消息/命令

+ (NSData *)packMsgDataSeqId:(NSString *)seqId msgType:(int32_t)msgTyep to:(NSString *)to contentType:(int32_t)contentType content:(NSString *)content extra:(NSString *)extra
{
    <!--上面定义了一个DataInputStream是解析服务器data的,这里的DataOutputStream是我们拼装data格式给服务器的类-->
    DataOutputStream *output = [[DataOutputStream alloc] init];
    <!--首先拼装header数据-->
    [output writeTcpProtocolHeaderWithCId:CmdType_MsgSend];
    <!--这里是消息数据结构-->
    SendRequest *msg = [[SendRequest alloc] init];
    msg.senderSeq = seqId;

    MsgSendBody *body = [[MsgSendBody alloc] init];
    body.mType = msgTyep;
    body.to = to;
    body.type = contentType;
    body.extra = extra;
    body.content = content;
    
    msg.msg = body;
    
    [output writeAESBytes:[msg data]];
    <!--将拼装好的内容转data准备使用scoket上传-->
    <!--[scoket writeData:data withTimeout:-1 tag:999];-->
    <!--使用writeData:withTimeout:tag:该方法上传-->
    return [output toByteArray];
}
<!--这里是如何拼装header数据 严格安装定义的格式: 先是header长度固定的 再是版本号 再是cid命令号 再试bid-->
-(void)writeTcpProtocolHeaderWithCId:(int16_t)cId
{
    [self writeShort:TCP_HEADER_LENGTH];
    [self writeShort:VERSION];
    [self writeShort:cId];
    [self writeInt:bid];
}
<!--举一个转换例子 其它依次类推-->
- (void)writeShort:(int16_t)v {
    int8_t ch[2];
    ch[0] = (v & 0x0ff00)>>8;
    ch[1] = (v & 0x0ff);
    [data appendBytes:ch length:2];
    length = length + 2;
}

2、Protobuf

Protocol Buffer是google 的一种数据交换的格式。它独立于语言,独立于平台。平常客户端与服务器都是使用JSON或者XML格式,但是在IM方面Protocol Buffer数据交换会更快,并且数据量更小。因为它是一种二进制数据传输格式。

在与服务器通信过程中,我们肯定要定义一些数据结构,然后再把这些定义的数据以二进制方式上传到服务器。这里就是Protocol Buffer起作用的时候了。这里有一个后缀为.proto的文件,其中定义的就是通信数据格式,之后我们会把这个文件转成OC.h .m文件。

例如:我们有一个Person数据格式,下面就是如何创建Person.proto文件

<!--指明proto的语法规则是proto2还是proto3-->
syntax = "proto3";
 <!--这里是我们定义的Person包含的数据-->
message Person
{
int32 age = 1;
string username = 2;
string phone = 3;
}
<!--这个文件里面还可以把需要的数据格式都定义好-->
message Student 
{

}

定义好Person.proto文件后就要把它转成OC的.h .m文件,它会以Person对象创建。转换命令如下:

protoc --proto_path=... --objc_out=... XXX.proto

其中proto_path是我们创建的proto文件所在目录,objc_outObjective-C文件输出路径,XXX.proto是我们创建的proto文件,可以一次转换多个proto文件,加在XXX.proto后面即可。

以上将Person.proto文件转换后会在输出文件夹内生成Person.pbobjc.h Person.pbobjc.m文件,将这两个文件放入到项目中,如果项目使用了ARC,要将.m(例子的Person.pbobjc.m)的Complier Flags设为-fno-objc-arc。(protobuf基于性能原因没有使用ARC)。

效果验证:


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

推荐阅读更多精彩内容