你好,歡迎來到IOS教程網

 Ios教程網 >> IOS使用技巧 >> IOS7技巧 >> IOS 詳解socket編程[oc]粘包、半包處理

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

編輯:IOS7技巧
小編推薦的這篇文章介紹了IOS 詳解socket編程[oc]粘包、半包處理,非常實用,有興趣的同學快來看看吧。

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

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

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

舉個栗子:

消息=消息頭+消息體。消息頭用於描述消息本身的基本信息,消息體則為消息的具體內容

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

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

消息體 =  len中描述的16字節長

所以這條消息的長度就是 26字節

可以看到,要想知道一條完整數據的邊界,關鍵就是消息頭中的len字段

假如我們現在接收到的數據是這樣的:

這個情況下即包含了粘包,也出現了半包的情況,三個數據包粘在了一起,最後一個數據包沒有接收完全,出現了半包的情況,看看代碼如何處理

 代碼如下復制代碼

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

{

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

  {

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

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

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

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

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

    {

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

      [self handleTcpResponseData:data];//處理包數據

      //從緩存中截掉處理完的數據,繼續循環

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

    }

    else//如果緩存中的數據長度不夠一個包的長度,則包不完整(處理半包,繼續讀取)

    {

      [_socket readDataWithTimeout:-1 buffer:_readBuf bufferOffset:_readBuf.length tag:0];//繼續讀取數據

      return;

    }

  }

  //緩存中數據都處理完了,繼續讀取新數據

  [_socket readDataWithTimeout:-1 buffer:_readBuf bufferOffset:_readBuf.length tag:0];//繼續讀取數據

}

  1. 上一頁:
  2. No
蘋果刷機越獄教程| IOS教程問題解答| IOS技巧綜合| IOS7技巧| IOS8教程
Copyright © Ios教程網 All Rights Reserved