php - tcp 粘包/拆包实例

栏目: PHP · 发布时间: 6年前

内容简介:将消息头的前N个字节固定为

tcp 长链接模式下,使用固定消息头长度的方式进行消息 拆包 ,解决 粘包 问题。

固定消息头协议

将消息头的前N个字节固定为 消息长度位 ,结合业务场景, 2bytes4bytes ,读取消息时先读取 消息长度位 ,即可按具体的 消息长度 读取 消息内容

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


以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

比特币

比特币

李钧、长铗 / 中信出版社 / 2014-1-1 / 39.00元

2009年,比特币诞生。比特币是一种通过密码编码,在复杂算法的大量计算下产生的电子货币。虽然是虚拟货币,比特币却引起了前所未有的全球关注热潮。 这一串凝结着加密算法与运算能力的数字不仅可以安全流通、换取实物,1比特币价值甚至曾高达8 000元人民币。有研究者认为比特币具备打破几千年来全球货币由国家垄断发行的可能性。在不经意间,比特币引起的金融新浪潮已悄然成型。 虚拟货币并不是新鲜事物,......一起来看看 《比特币》 这本书的介绍吧!

JSON 在线解析
JSON 在线解析

在线 JSON 格式化工具

XML、JSON 在线转换
XML、JSON 在线转换

在线XML、JSON转换工具

html转js在线工具
html转js在线工具

html转js在线工具