您好,登錄后才能下訂單哦!
今天小編給大家分享一下php的tcp粘包和拆包怎么實現的相關知識點,內容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。
tcp 長鏈接模式下,使用固定消息頭長度的方式進行消息 拆包 ,解決 粘包 問題。 固定消息頭協議
將消息頭的前 N 個字節固定為 消息長度位 ,結合業務場景, 2bytes 或 4bytes ,讀取消息時先讀取 消息長度位 ,即可按具體的 消息長度 讀取 消息內容 。
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
拆包
日常工作中經常遇到的 tcp 場景可能是 短連接單個消息 的模式,客戶端發送一條消息后便關閉連接,服務端循環讀取到 EOF 即可得到一條完整的消息。但如果是 短連接多個消息 或 長鏈接模式 下,就可能會發生粘包,客戶端不關閉服務端無法通過 EOL 確定消息讀取完畢的問題。這就需要定義協議和拆包。<?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;
以上就是“php的tcp粘包和拆包怎么實現”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。