首頁  >  文章  >  後端開發  >  分享一個php的tcp黏包/拆包例子

分享一個php的tcp黏包/拆包例子

藏色散人
藏色散人轉載
2022-11-29 16:52:246518瀏覽

本文跟大家介紹一個例子,有關PHP的tcp 黏包/拆包,希望對需要的朋友有所幫助~

分享一個php的tcp黏包/拆包例子

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

將訊息頭的前N 個位元組固定為訊息長度位,結合業務場景, 2bytes 或4bytes ,讀取訊息時先讀取訊息長度位,即可按具體的訊息長度讀取訊息內容。 【推薦學習:PHP影片教學

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的tcp黏包/拆包例子的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文轉載於:learnku.com。如有侵權,請聯絡admin@php.cn刪除