固定头协议

将头的前N个字节固定为长度位 ,结合业务场景,2bytes 或4bytes ,读取消息时先读取消息长度位 ,即可按详细的长度 读取消息内容 。

pack/unpack 可以打包数值至二进制 /解包二进制至数值 ,详细的模式可以参考pack/unpack 详细用法,这里我们选用固定头长度为2bytes 来表示体长度 ,最大能表示2^16 - 1 长度的体,不足你就上4bytes 好了。

phptcp高位序带你懂得php  tcp 粘包/拆包实例 Bootstrap

组包

<?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 确定读取完毕的问题。
这就须要定义协议和拆包