首页 > 代码库 > AsyncSocket长连接粘包问题解决方案

AsyncSocket长连接粘包问题解决方案

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


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