一、简介

工程中使用长连接来和服务器进行通讯,因此,我们的协议通过 指定前两个字节为数据长度 来区分数据包

app这边数据有两种传输形式:

1.app主动请求所需要的数据;

2.app异步接收来自服务端的推送消息,也就是app这边没有请求,服务端主动发送数据到app客户端;

二、引发粘包的原因

整个app运行期间,它们都是在同一个连接上完成的数据传输,因此会出现以下的问题:

1.服务器数据传输过快,出现粘包的问题,例如

1.1服务端一次发来多个推送消息;

1.2网络不稳定,客户端连续发送多个请求客户端一次接收到全部答复;

2.客户端的一个请求报文,服务端的应答报文数据过大,到IP层需要进行分片,因此客户端这边就会出现几次才接收到完整的数据的情况;

首先有以下4个方法需要介绍

/**
 **实例方法
 **调用此方法以后,当套接字接收缓冲区有可用字节的时候,会触发onSocket:didReadData:withTag:委托方法,此时接收到的数据会出现上面说到的问题
 */
- (void)readDataWithTimeout:(NSTimeInterval)timeout 
                        tag:(long)tag;

/**
 **实例方法
 **调用此方法以后,当套接字接收缓冲区有length长度的可用字节的时候,会触发onSocket:didReadData:withTag:委托方法,此时接收到固定长度的数据,这个固定长度就是length给出的值,当length的长度大于接收缓冲区数据的长度的时候,就会等待,直到接收到length长度的数据的时候才会触发以上委托方法的调用
 */
- (void)readDataToLength:(NSUInteger)length 
             withTimeout:(NSTimeInterval)timeout 
                     tag:(long)tag;


/**
 **实例方法
 **此方法功能同上,只是多了几个参数,buffer是你将接收的数据写到的地方,offset是写到buffer中的偏移位置
 */
- (void)readDataToLength:(NSUInteger)length
             withTimeout:(NSTimeInterval)timeout
                  buffer:(NSMutableData *)buffer
            bufferOffset:(NSUInteger)offset
                     tag:(long)tag;


/**
 **委托方法
 **此方法上面已经说到
 */
- (void)onSocket:(AsyncSocket *)sock 
     didReadData:(NSData *)_data 
         withTag:(long)tag;

三、解决方法

客户端每次发送请求以后,首先只接收两个字节的长度字节,如下:

[sendSocket readDataToLength:2 withTimeout:set.timeout tag:tag];  
[sendSocket writeData:data withTimeout:set.timeout tag:tag];

然后当有可用字节到达套接字接收缓冲区的时候触发以下委托方法,我们在里面做如下处理,这样既解决了粘包的问题,也解决了数据过大,多次接收完整的问题;

- (void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)_data withTag:(long)tag  
{  
    SettingData* set = [SettingData shareSettingData];  

    if (respondData == nil) {  
        respondData = [[NSMutableData alloc]init];  
        respondDataLen = [RequestUnit respondMessageLengthWithData:_data];  
        [sock readDataToLength:respondDataLen withTimeout:set.timeout tag:tag];  
        return;  
    }  
    [respondData appendData:[RequestUnit respondBytesToUTF8Data:_data]];  
    [self parserData:respondData withTag:tag];  
}