美文网首页
iOS自建IM相关

iOS自建IM相关

作者: joeal | 来源:发表于2021-05-26 20:51 被阅读0次

一、涉及到的第三方库

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);
 
}

相关文章

  • iOS自建IM相关

    一、涉及到的第三方库 1、GCDAsyncSocket GCDAsyncSocket是一个封装好的,帮助开发者完成...

  • 自建iOS测试服务实践

    背景 没有自建iOS测试服务,iOS测试大多依赖蒲公英测试平台或者fir.im,若没有遇到什么问题,继续使用第三方...

  • fir.im Weekly - 揭秘 iOS 面向协议编程

    本期 fir.im Weekly 重点推荐关于 iOS 面向协议编程相关文章,还有 iOS 多线程安全、Swift...

  • iOS 自建IM即时通讯

    一、原由 其实现在有很多大厂IM的第三方,云信、融云、TIM,本人都有在项目中集成过,第三方好处就是你不用太多关注...

  • XMPPFramework使用记录(一)

    前言 最近公司需要我们使用XMPP协议,实现一个简单的IM模块。在此之前没有接触过IM相关技术,仅了解iOS可以通...

  • iOS端IM开发从入门到填坑

    iOS端IM开发从入门到填坑 iOS端IM开发从入门到填坑

  • iM

    ios IM 架构设计 ios IM 网络层架构 ios 侧滑与右滑返回手势http://blog.csdn.n...

  • 金易联iOS SDK 集成文档

    一、iOS SDK 介绍 FinoChat SDK 为用户开发 IM 相关的应用提供的一套完善的开发框架。包括以下...

  • xcode 使用笔记

    最近做个 IM 相关的项目,为了搞个开源的 IOS 版本,因此开启了 XCode 开发使用 遇见了如下几点问题 环...

  • iOS高级-Python自动化打企业包上传fir.im发送邮件

    ios Python自动化打包上传fir.im发送邮件通知相关人 一、fir命令安装,终端内,执行: 二、pyth...

网友评论

      本文标题:iOS自建IM相关

      本文链接:https://www.haomeiwen.com/subject/pyrlsltx.html