请教各位一个问题,求解答:为什么 PHP unserialize 会有内存泄露怎么回事哈啊?

最近写一个PHP服务,PHP版本为swoolePHP环境二进制版7.x,其中部分逻辑涉及到把PHP变量写入到文件中,后续处理的时候读出来,一开始使用serialize和unserialize实现,但是运行不久就直接超内存了,最后发现是unserialize造成的内存泄露
例子:
测试数据:serialize序列化字符串
[www@chengqmtest]$catserialize_str
a:1:{s:4:"test";s:1024:"1111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111";}
测试数据:json序列化字符串
[www@chengqmtest]$catjson_str
{"test":"1111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111"}
然后是PHP代码,分别反序列化这两个文件的内容,前后输出内存使用情况
echoPHP_EOL;
echo'beforeunserialize:'.memory_get_usage().PHP_EOL;
$serialize_str=file_get_contents('./serialize_str');
$array=unserialize($serialize_str);
unset($serialize_str);
unset($array);
echo'afterunsetargs:'.memory_get_usage().PHP_EOL;
echoPHP_EOL;
echo'beforedecode_json:'.memory_get_usage().PHP_EOL;
$json_str=file_get_contents('./json_str');
$array=json_decode($serialize_str,True);
unset($json_str);
unset($array);
echo'afterunsetargs:'.memory_get_usage().PHP_EOL;
结果
[www@chengqmtest]$php-swooletest.php
beforeunserialize:625048
afterunsetargs:625192
beforedecode_json:625192
afterunsetargs:625192
unserialize的某部分内存没有释放,json_decode的已经释放了
问题:为什么会有部分内存没有释放,是BUG还是PHP的某项特性
-------------------问题更新---------------------
上面代码写错变量,重新测一下,这次没有读取文件,json_decode放在前面
echoPHP_EOL;
echo'beforedecode_json:'.memory_get_usage().PHP_EOL;
$json_str='{"test":"1111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111"}';
$array=json_decode($json_str,True);
unset($json_str);
unset($array);
echo'afterunsetargs:'.memory_get_usage().PHP_EOL;
echoPHP_EOL;
echo'beforeunserialize:'.memory_get_usage().PHP_EOL;
$serialize_str='a:1:{s:4:"test";s:1024:"1111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111";}';
$array=unserialize($serialize_str);
unset($serialize_str);
unset($array);
echo'afterunsetargs:'.memory_get_usage().PHP_EOL;
[root@chengqmtest]#php-swooletest.php
beforedecode_json:627504
afterunsetargs:627504
beforeunserialize:627504
afterunsetargs:627536
结果发现,unserialize后确实有一点内存没有释放(后来发现,是安装的PHP有问题)
然后就是测试file_get_contents对内存的影响
echoPHP_EOL;
echo'beforegetfile:'.memory_get_usage().PHP_EOL;
$serialize_str=file_get_contents('./serialize_str');
unset($serialize_str);
system('sync&&echo3>/proc/sys/vm/drop_caches');
echo'afterunsetargs:'.memory_get_usage().PHP_EOL;
[root@chengqmtest]#php-swooletest.php
beforegetfile:624304
afterunsetargs:624416
结果发现,file_get_contents也会造成内存增长
智慧大石
浏览 301回答 2
2回答

拉丁的传说

valgrind跑了下,应该不算是内存泄漏跑这个也是两次都不一样,第三次就一样了……
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

JavaScript