内容简介:将消息头的前N个字节固定为
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
确定消息读取完毕的问题。这就需要定义协议和拆包。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持 码农网
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
翻转课堂的可汗学院
萨尔曼·可汗(Salman Khan) / 刘婧 / 浙江人民出版社 / 2014-4-1 / 49.00元
MIT和哈佛毕业的高材生缘何放弃金融分析师工作投身教育事业?YouTube上的“可汗学院频道”至今共吸引了163.3万订阅者,观看次数超过3.55亿次,它为什么如此大受欢迎?创始人萨尔曼·可汗阐述属于未来的教育理念——让地球上的任何人都能随时随地享受世界一流的免费教育! 现行教育模式已有200余年历史,可汗认为,在互联网蓬勃发展、社交网络盛况空前的时代,免费、灵活、适合个体、全球共享的教育才......一起来看看 《翻转课堂的可汗学院》 这本书的介绍吧!