IOS 詳解socket編程[oc]粘包、半包處理
IOS 詳解socket編程[oc]粘包、半包處理
在做socket編程時(shí),如果是做tcp連接,那就不可避免的會(huì)遇到粘包與半包的問題,粘包就是多組數(shù)據(jù)被一并接收了,粘在了一起,無法做劃分;半包就是有數(shù)據(jù)接收不完整,無法處理。要解決粘包、半包的問題,一般在設(shè)計(jì)數(shù)據(jù)(消息)格式時(shí)會(huì)約定好一個(gè)字段專門用于描述數(shù)據(jù)包的長(zhǎng)度,這樣就使數(shù)據(jù)有了邊界,依靠這個(gè)邊界,就能把每組數(shù)據(jù)劃分出來,數(shù)據(jù)不完整時(shí)也能獲知數(shù)據(jù)的缺失。
(當(dāng)然也可以把數(shù)據(jù)設(shè)計(jì)成定長(zhǎng)數(shù)據(jù),但這樣不夠靈活;或者用\n,\r這類字符作為數(shù)據(jù)劃分依據(jù),但不直觀、不明確,同時(shí)也不靈活)
舉個(gè)栗子:
消息=消息頭+消息體。消息頭用于描述消息本身的基本信息,消息體則為消息的具體內(nèi)容

如上圖所示,假如我們的一個(gè)消息是這么定義的
消息頭 = msgId(4B)+version(2B)+len(4B),共占用10字節(jié)
消息體 = len中描述的16字節(jié)長(zhǎng)
所以這條消息的長(zhǎng)度就是 26字節(jié)
可以看到,要想知道一條完整數(shù)據(jù)的邊界,關(guān)鍵就是消息頭中的len字段
假如我們現(xiàn)在接收到的數(shù)據(jù)是這樣的:

這個(gè)情況下即包含了粘包,也出現(xiàn)了半包的情況,三個(gè)數(shù)據(jù)包粘在了一起,最后一個(gè)數(shù)據(jù)包沒有接收完全,出現(xiàn)了半包的情況,看看代碼如何處理
- (void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag
{
while (_readBuf.length >= 10)//因?yàn)轭^部固定10個(gè)字節(jié),數(shù)據(jù)長(zhǎng)度至少要大于10個(gè)字節(jié),我們才能得到完整的消息描述信息
{
NSData *head = [_readBuf subdataWithRange:NSMakeRange(0, 10)];//取得頭部數(shù)據(jù)
NSData *lengthData = [head subdataWithRange:NSMakeRange(6, 4)];//取得長(zhǎng)度數(shù)據(jù)
NSInteger length = [[[NSString alloc] initWithData:lengthData encoding:NSUTF8StringEncoding] integerValue];//得出內(nèi)容長(zhǎng)度
NSInteger complateDataLength = length + 10;//算出一個(gè)包完整的長(zhǎng)度(內(nèi)容長(zhǎng)度+頭長(zhǎng)度)
if (_readBuf.length >= complateDataLength)//如果緩存中數(shù)據(jù)夠一個(gè)整包的長(zhǎng)度
{
NSData *data = [_readBuf subdataWithRange:NSMakeRange(0, complateDataLength)];//截取一個(gè)包的長(zhǎng)度(處理粘包)
[self handleTcpResponseData:data];//處理包數(shù)據(jù)
//從緩存中截掉處理完的數(shù)據(jù),繼續(xù)循環(huán)
_readBuf = [NSMutableData dataWithData:[_readBuf subdataWithRange:NSMakeRange(complateDataLength, _readBuf.length - complateDataLength)]];
}
else//如果緩存中的數(shù)據(jù)長(zhǎng)度不夠一個(gè)包的長(zhǎng)度,則包不完整(處理半包,繼續(xù)讀取)
{
[_socket readDataWithTimeout:-1 buffer:_readBuf bufferOffset:_readBuf.length tag:0];//繼續(xù)讀取數(shù)據(jù)
return;
}
}
//緩存中數(shù)據(jù)都處理完了,繼續(xù)讀取新數(shù)據(jù)
[_socket readDataWithTimeout:-1 buffer:_readBuf bufferOffset:_readBuf.length tag:0];//繼續(xù)讀取數(shù)據(jù)
}
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
iOS開發(fā)Masonry與Frame布局差異示例詳解
這篇文章主要為大家介紹了iOS開發(fā)Masonry與Frame布局差異示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
iOS 9 更新之Safari廣告攔截器(Content Blocker)開發(fā)教程
這篇文章主要介紹了iOS 9 更新之Safari廣告攔截器(Content Blocker)開發(fā)教程的相關(guān)資料,需要的朋友可以參考下2015-08-08
ios NSNotificationCenter通知的簡(jiǎn)單使用
這篇文章主要介紹了ios NSNotificationCenter通知的簡(jiǎn)單使用,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-06-06
iOS 導(dǎo)航欄無縫圓滑的隱藏 Navigationbar實(shí)例代碼
本文通過實(shí)例代碼給大家介紹了iOS 導(dǎo)航欄無縫圓滑的隱藏 Navigationbar的效果,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2017-11-11
iOS 數(shù)據(jù)結(jié)構(gòu)之?dāng)?shù)組的操作方法
這篇文章主要介紹了iOS 數(shù)據(jù)結(jié)構(gòu)之?dāng)?shù)組的操作方法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2018-07-07
iOS 控制器自定義動(dòng)畫跳轉(zhuǎn)方法(模態(tài)跳轉(zhuǎn))
下面小編就為大家分享一篇iOS 控制器自定義動(dòng)畫跳轉(zhuǎn)方法(模態(tài)跳轉(zhuǎn)),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-01-01

