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