php - tcp 黏包/拆包範例

2020-07-16 10:05:40

tcp 長連結模式下,使用固定訊息頭長度的方式進行訊息拆包,解決黏包問題。

固定訊息頭協定

將訊息頭的前N個位元組固定為訊息長度位,結合業務場景,2bytes 或 4bytes,讀取訊息時先讀取訊息長度位,即可按具體的訊息長度讀取訊息內容。

pack/unpack 可以打包數值至二進位制/解包二進位制至數值,具體的模式可以參考 pack/unpack 詳細用法,這裡我們選用固定頭長度為2bytes來表示訊息體長度,最大能表示2^16 - 1長度的訊息體,不夠你就上4bytes好了。

組包

<?php// msg protocol// | ---- dataLen ---- | data |// | - fixed 2bytes  - |// 模擬用戶端連續傳送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 位元組序為 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;

日常工作中經常遇到的tcp場景可能是短連線單個訊息的模式,用戶端傳送一條訊息後便關閉連線,伺服器端迴圈讀取到EOF即可得到一條完整的訊息。但如果是短連線多個訊息或長連結模式下,就可能會發生黏包,用戶端不關閉伺服器端無法通過EOL確定訊息讀取完畢的問題。這就需要定義協定和拆包。

更多PHP相關技術文章,請存取PHP教學欄目進行學習!

以上就是php - tcp 黏包/拆包範例的詳細內容,更多請關注TW511.COM其它相關文章!