您好,登錄后才能下訂單哦!
這篇文章主要介紹php - tcp 粘包/拆包的案例分析,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!
tcp 長鏈接模式下,使用固定消息頭長度的方式進行消息拆包,解決粘包問題。
固定消息頭協(xié)議
將消息頭的前N個字節(jié)固定為消息長度位,結合業(yè)務場景,2bytes 或 4bytes,讀取消息時先讀取消息長度位,即可按具體的消息長度讀取消息內容。
pack/unpack 可以打包數(shù)值至二進制/解包二進制至數(shù)值,具體的模式可以參考 pack/unpack 詳細用法,這里我們選用固定頭長度為2bytes來表示消息體長度,最大能表示2^16 - 1長度的消息體,不夠你就上4bytes好了。
組包
<?php// msg protocol// | ---- dataLen ---- | data |// | - fixed 2bytes - |// 模擬客戶端連續(xù)發(fā)送2條消息$foo = "hello world"; $bar = "i am sqrt_cat"; $package = "";// 使用 n 打包 固定2bytes$fooLenn = pack("n", strlen($foo)); $package = $fooLenn . $foo; $barLenn = pack("n", strlen($bar)); $package .= $barLenn . $bar;
粘包
// send// 傳輸 $package 由 $foo $bar 兩條消息組成 模擬粘包場景 // receive
拆包
<?php // 解析第1條消息 取前 2bytes 按 n 解包 $fooLen = unpack("n", substr($package, 0, 2))[1]; // 使用包消息體長度定義讀取消息體 // 從第 3byte 開始讀 前 2bytes表示長度 $foo = substr($package, 2, $fooLen); echo $foo . PHP_EOL; // 解析第2條消息 取前 2bytes 按 n 解包 // 0 ~ (2 + fooLen) - 1 字節(jié)序為 fooLen . foo // (2 + fooLen) ~ (2 + fooLen) + 2 - 1 為 barLen $barLen = unpack("n", substr($package, (2 + $fooLen), 2))[1]; $bar = substr($package, (2 + $fooLen) + 2, $barLen); echo $bar . PHP_EOL;
日常工作中經(jīng)常遇到的tcp場景可能是短連接單個消息的模式,客戶端發(fā)送一條消息后便關閉連接,服務端循環(huán)讀取到EOF即可得到一條完整的消息。但如果是短連接多個消息或長鏈接模式下,就可能會發(fā)生粘包,客戶端不關閉服務端無法通過EOL確定消息讀取完畢的問題。這就需要定義協(xié)議和拆包。
以上是php - tcp 粘包/拆包的案例分析的所有內容,感謝各位的閱讀!希望分享的內容對大家有幫助,更多相關知識,歡迎關注億速云行業(yè)資訊頻道!
免責聲明:本站發(fā)布的內容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內容。