Swoole 4.5.6 支持零拷贝 JSON 或 PHP 反序列化
在最新的4.5.6
开发分支中,底层增加了2
个特殊的函数:
swoole_substr_json_decode
swoole_substr_unserialize
这里为什么要增加这两个函数呢?有这样一种场景。使用Swoole\Server
实现RPC
服务,在EOF
协议或长度协议通信方式下,一个包可能有3
部分组成。
$packet = $header + $body + $footer
通常$header
和$footer
比较小,而$body
比较大,$body
可能会使用JSON
或PHP
序列化格式。如果要解析$body
数据,那么就需要先进行substr
得到$body
的字符串格式数据,再进行json_decode
和unserialize
操作。
这会引起一次内存拷贝,$body_str = substr($packet, $header_length)
的过程会创建一个临时字符串变量,再反序列化操作$body = json_decode($body_str)
之后,这个变量就会被释放。
// 先进行 substr,这时会产生内存拷贝,从 $packet 复制数据到 $body_str
$body_str = substr($packet, 4, strlen($packet) - 4 - 2);
// 反序列化之后 $body_str 这块内存不再使用,会在函数退出时释放
$body = json_decode($body_str, true);
使用新增的两个函数就可以将substr
和反序列化
操作合二为一。减少一次内存拷贝,从而提高性能。
$body = swoole_substr_json_decode($packet, $header_length);
$body = swoole_substr_unserialize($packet, $header_length);
压测
<?php
error_reporting(E_ALL);
$a['hello'] = base64_encode(random_bytes(1000));
$a['world'] = 'hello';
$a['int'] = rand(1, 999999);
$a['list'] = ['a,', 'b', 'xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx'];
$val = serialize($a);
$str = pack('N', strlen($val)).$val."rn";
$n = 100000;
$s = microtime(true);
while($n--) {
$l = strlen($str) - 6;
// var_dump(unserialize(substr($str, 4, $l)));
var_dump(swoole_substr_unserialize($str, 4, $l));
}
echo "cost: ".(microtime(true)-$s)."n";
使用swoole_substr_unserialize
与substr + unserialize
相比,性能提升了12%
aaa@qq.com:~/workspace/debug$ php s.php
cost: 2.2559139728546
aaa@qq.com:~/workspace/debug$ php s.php
cost: 1.9821600914001
Swoole 4.5.6 支持零拷贝 JSON 或 PHP 反序列化
很多PHPer在进阶的时候总会遇到一些问题和瓶颈,业务代码写多了迷茫没方向,不知道该从哪儿入手去提升自己。→→管理整理了一些资料,有 腾讯 等一线大厂进阶知识体系 可供参考(相关学习资料以及笔面试题)
覆盖各个技术栈:分布式架构、高可扩展、高性能、高并发、服务器性能调优、TP6,laravel,YII2,Redis,Swoole、Swoft、Kafka、Mysql优化、shell脚本、Docker、微服务、Nginx等多个知识点高级进阶干货欢迎加入我的官方群啊点击此处
上一篇: leetcode 652
下一篇: P1553 数字反转(升级版)