tcp
長連接模式下,使用固定消息頭長度的方式進行消息拆包
,解決粘包
問題。php
將消息頭的前N個字節固定爲消息長度位
,結合業務場景,2bytes
或 4bytes
,讀取消息時先讀取消息長度位
,便可按具體的消息長度
讀取消息內容
。segmentfault
pack/unpack
能夠打包數值至二進制
/解包二進制至數值
,具體的模式能夠參考 pack/unpack 詳細用法,這裏咱們選用固定頭長度爲2bytes
來表示消息體長度
,最大能表示2^16 - 1
長度的消息體,不夠你就上4bytes
好了。tcp
<?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
肯定消息讀取完畢的問題。這就須要定義協議和拆包。spa