猿问

萌新求教!为什么 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也会造成内存增长
海绵宝宝撒
浏览 291回答 2
2回答

缥缈止盈

首先,倒数第四行$array=json_decode($serialize_str,True);似乎是写错了,应该是$array=json_decode($json_str,true);吧。。然后不知道楼主有没有尝试将json段放在文件前面,将serialize段放在文件后面,这样去执行的话你就会发现似乎是json段执行之后有内存没有释放了。。所以,这里其实不是unserialize造成的内存泄漏。。这边造成有一部分内存被莫名吃掉应该是file_get_contents造成的,楼主单独运行一次file_get_contents就可以验证一下了。。最后说说楼主遇到的问题:在Linux中,频繁地存取文件会产生大量的文件缓存,这是操作系统的机制,不是PHP的机制,楼主可查看下buff/cache的大小,内存有可能都被cache吃掉了。PHP的GC机制中,自动回收的只有array和obj两类,因为这两类会产生循环引用,造成真正的内存泄漏。而file_get_contents这种不完全归还内存的情况个人猜测(未验证,纯属猜测,轻喷)是内存映射机制造成的小内存碎片没有回收。。

斯蒂芬大帝

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

相关分类

JavaScript
我要回答