一、简介
工程中使用长连接来和服务器进行通讯,因此,我们的协议通过 指定前两个字节为数据长度 来区分数据包
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];
}