Java網(wǎng)絡(luò)編程基礎(chǔ)之TCP粘包拆包

TCP是個(gè)“流”協(xié)議,所謂流袖肥,就是沒(méi)有界限的一串?dāng)?shù)據(jù)咪辱。大家可以想象河里的流水,他們是連成一片的椎组,其間并沒(méi)有分界線油狂。TCP底層并不了解上層業(yè)務(wù)數(shù)據(jù)的具體含義,他會(huì)根據(jù)TCP緩沖區(qū)的實(shí)際情況進(jìn)行包的劃分,所以在業(yè)務(wù)上認(rèn)為专筷,一個(gè)完整的包可能會(huì)被TCP拆分成多個(gè)包進(jìn)行發(fā)送弱贼,也有可能把多個(gè)小的包封裝成一個(gè)大的數(shù)據(jù)包發(fā)送。這就是TCP所謂的拆包和粘包的問(wèn)題磷蛹。

一吮旅、TCP粘包/拆包問(wèn)題說(shuō)明

我們可以通過(guò)圖解對(duì)TCP粘包和拆包問(wèn)題進(jìn)行說(shuō)明,粘包問(wèn)題如圖味咳。

image

假設(shè)客戶端分別發(fā)送了兩個(gè)數(shù)據(jù)包D1和D2給服務(wù)端庇勃,由于服務(wù)端一次讀取到的字節(jié)數(shù)是不確定的,故可能存在以下4中情況槽驶。

  • 服務(wù)端分兩次讀取到了兩個(gè)獨(dú)立的數(shù)據(jù)包责嚷,分別是D1和D2,沒(méi)有粘包和拆包捺檬。
  • 服務(wù)端一次接收到了兩個(gè)數(shù)據(jù)包再层,D1和D2粘在一起,被稱為TCP粘包
  • 服務(wù)端分兩次讀取到了兩個(gè)數(shù)據(jù)包堡纬,第一次讀取到了完整的D1包和D2包的部分內(nèi)容聂受,第二次讀取到了D2包的剩余內(nèi)容,這被稱為TCP拆包烤镐。
  • 服務(wù)端分兩次讀取到了兩個(gè)數(shù)據(jù)包蛋济,第一次讀取到了D1包的部分內(nèi)容D1_1,第二次讀取到了D1包的剩余內(nèi)容D1_2和D2包的整包炮叶。

如果此時(shí)服務(wù)端TCP接收滑窗非常小碗旅,而數(shù)據(jù)包D1和D2比較大,很有可能會(huì)發(fā)生第五種可能镜悉,即服務(wù)端分多次才能將D1和D2包接收完全祟辟,期間發(fā)生多次拆包。

二侣肄、TCP粘包/拆包發(fā)生的原因

問(wèn)題產(chǎn)生的原因有三個(gè)旧困,分別如下。

  • 應(yīng)用程序write寫入的字節(jié)大小大于套接口發(fā)送緩沖區(qū)大小稼锅。
  • 進(jìn)行MSS大小的TCP分段吼具。
  • 以太網(wǎng)幀的payload大于MTU進(jìn)行IP分片。

三矩距、粘包問(wèn)題的解決策略

由于底層的TCP無(wú)法理解上層的業(yè)務(wù)數(shù)據(jù)拗盒,所以在底層是無(wú)法保證數(shù)據(jù)包不被拆分和重組的,這個(gè)問(wèn)題只能通過(guò)上層的應(yīng)用協(xié)議棧設(shè)計(jì)來(lái)解決锥债,根據(jù)業(yè)界的主流協(xié)議的解決方案陡蝇,可以歸納如下痊臭。

  • 消息定長(zhǎng),例如每個(gè)報(bào)文的大小為固定長(zhǎng)度200字節(jié)毅整,如果不夠趣兄,空位補(bǔ)空格
  • 在包尾增加回車換行符進(jìn)行分割,例如FTP協(xié)議
  • 將消息分為消息頭和消息體悼嫉,消息頭中包含表示消息總長(zhǎng)度(或者消息體長(zhǎng)度)的字段,通常涉及思路為消息頭的第一個(gè)字段使用int32來(lái)表示消息的總長(zhǎng)度
  • 更復(fù)雜的應(yīng)用層協(xié)議拼窥。
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末戏蔑,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子鲁纠,更是在濱河造成了極大的恐慌总棵,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,657評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件改含,死亡現(xiàn)場(chǎng)離奇詭異情龄,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)捍壤,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門骤视,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人鹃觉,你說(shuō)我怎么就攤上這事专酗。” “怎么了盗扇?”我有些...
    開(kāi)封第一講書人閱讀 164,057評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵祷肯,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我疗隶,道長(zhǎng)佑笋,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書人閱讀 58,509評(píng)論 1 293
  • 正文 為了忘掉前任斑鼻,我火速辦了婚禮蒋纬,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘卵沉。我一直安慰自己颠锉,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,562評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布史汗。 她就那樣靜靜地躺著琼掠,像睡著了一般。 火紅的嫁衣襯著肌膚如雪停撞。 梳的紋絲不亂的頭發(fā)上瓷蛙,一...
    開(kāi)封第一講書人閱讀 51,443評(píng)論 1 302
  • 那天悼瓮,我揣著相機(jī)與錄音,去河邊找鬼艰猬。 笑死横堡,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的冠桃。 我是一名探鬼主播命贴,決...
    沈念sama閱讀 40,251評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼食听!你這毒婦竟也來(lái)了胸蛛?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書人閱讀 39,129評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤樱报,失蹤者是張志新(化名)和其女友劉穎葬项,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體迹蛤,經(jīng)...
    沈念sama閱讀 45,561評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡民珍,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,779評(píng)論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了盗飒。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片嚷量。...
    茶點(diǎn)故事閱讀 39,902評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖箩兽,靈堂內(nèi)的尸體忽然破棺而出津肛,到底是詐尸還是另有隱情,我是刑警寧澤汗贫,帶...
    沈念sama閱讀 35,621評(píng)論 5 345
  • 正文 年R本政府宣布身坐,位于F島的核電站,受9級(jí)特大地震影響落包,放射性物質(zhì)發(fā)生泄漏部蛇。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,220評(píng)論 3 328
  • 文/蒙蒙 一咐蝇、第九天 我趴在偏房一處隱蔽的房頂上張望涯鲁。 院中可真熱鬧,春花似錦有序、人聲如沸抹腿。這莊子的主人今日做“春日...
    開(kāi)封第一講書人閱讀 31,838評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)警绩。三九已至,卻和暖如春盅称,著一層夾襖步出監(jiān)牢的瞬間肩祥,已是汗流浹背后室。 一陣腳步聲響...
    開(kāi)封第一講書人閱讀 32,971評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留混狠,地道東北人岸霹。 一個(gè)月前我還...
    沈念sama閱讀 48,025評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像将饺,于是被迫代替她去往敵國(guó)和親贡避。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,843評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容