ios開發(fā)中的包的處理

IOS 詳解socket編程[oc]粘包、半包處理

在做socket編程時,如果是做tcp連接,那就不可避免的會遇到粘包與半包的問題,粘包就是多組數(shù)據(jù)被一并接收了,粘在了一起,無法做劃分;半包就是有數(shù)據(jù)接收不完整,無法處理。要解決粘包、半包的問題,一般在設(shè)計數(shù)據(jù)(消息)格式時會約定好一個字段專門用于描述數(shù)據(jù)包的長度,這樣就使數(shù)據(jù)有了邊界,依靠這個邊界,就能把每組數(shù)據(jù)劃分出來,數(shù)據(jù)不完整時也能獲知數(shù)據(jù)的缺失。

(當(dāng)然也可以把數(shù)據(jù)設(shè)計成定長數(shù)據(jù),但這樣不夠靈活;或者用\n,\r這類字符作為數(shù)據(jù)劃分依據(jù),但不直觀、不明確,同時也不靈活)

舉個栗子:


消息=消息頭+消息體。消息頭用于描述消息本身的基本信息,消息體則為消息的具體內(nèi)容

如上圖所示,假如我們的一個消息是這么定義的

消息頭 = msgId(4B)+version(2B)+len(4B),共占用10字節(jié)

消息體 =? len中描述的16字節(jié)長

所以這條消息的長度就是 26字節(jié)

可以看到,要想知道一條完整數(shù)據(jù)的邊界,關(guān)鍵就是消息頭中的len字段

假如我們現(xiàn)在接收到的數(shù)據(jù)是這樣的:


這個情況下即包含了粘包,也出現(xiàn)了半包的情況,三個數(shù)據(jù)包粘在了一起,最后一個數(shù)據(jù)包沒有接收完全,出現(xiàn)了半包的情況,看看代碼如何處理

- (void)onSocket:(AsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag

{

while (_readBuf.length >= 10)//因為頭部固定10個字節(jié),數(shù)據(jù)長度至少要大于10個字節(jié),我們才能得到完整的消息描述信息

{

NSData *head = [_readBuf subdataWithRange:NSMakeRange(0, 10)];//取得頭部數(shù)據(jù)

NSData *lengthData = [head subdataWithRange:NSMakeRange(6, 4)];//取得長度數(shù)據(jù)

NSInteger length = [[[NSString alloc] initWithData:lengthData encoding:NSUTF8StringEncoding] integerValue];//得出內(nèi)容長度

NSInteger complateDataLength = length + 10;//算出一個包完整的長度(內(nèi)容長度+頭長度)

if (_readBuf.length >= complateDataLength)//如果緩存中數(shù)據(jù)夠一個整包的長度

{

NSData *data = [_readBuf subdataWithRange:NSMakeRange(0, complateDataLength)];//截取一個包的長度(處理粘包)

[self handleTcpResponseData:data];//處理包數(shù)據(jù)

//從緩存中截掉處理完的數(shù)據(jù),繼續(xù)循環(huán)

_readBuf = [NSMutableData dataWithData:[_readBuf subdataWithRange:NSMakeRange(complateDataLength, _readBuf.length - complateDataLength)]];

}

else//如果緩存中的數(shù)據(jù)長度不夠一個包的長度,則包不完整(處理半包,繼續(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ù)

}

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容