【PHP内存泄漏案例】PHP对象递归引用造成内存泄漏
【案例一】
作者:老王
如果PHP对象存在递归引用,就会出现内存泄漏。这个Bug在PHP里已经存在很久很久了,先让我们来重现这个Bug,代码如下:
<?php class Foo { function __construct() { $this->bar = new Bar($this); } } class Bar { function __construct($foo) { $this->foo = $foo; } } for ($i = 0; $i < 100; $i++) { $obj = new Foo(); unset($obj); echo memory_get_usage(), " "; } ?>
运行以上代码,你会发现,内存使用量本应该不变才对,可实际上却是不断增加,unset没有完全生效。
现在的开发很多都是基于框架进行的,应用里存在复杂的对象关系,那么就很可能会遇到这样的问题,下面看看有什么权宜之计:
<?php class Foo { function __construct() { $this->bar = new Bar($this); } function __destruct() { unset($this->bar); } } class Bar { function __construct($foo) { $this->foo = $foo; } } for ($i = 0; $i < 100; $i++) { $obj = new Foo(); $obj->__destruct(); unset($obj); echo memory_get_usage(), " "; } ?>
办法有些丑陋,不过总算是对付过去了。幸运的是这个Bug在PHP5.3的CVS代码中已经被修复了。
参考链接:http://bugs.php.net/bug.php?id=33595
【案例二】
作者:贵贵的博客
出现了一个PHP脚本占用系统内存50%
开始认为是哪个变量PHP没有自动释放,查看代码,里面有个类的函数是static的,加了设置为Null和unset后无效果。
用memory_get_usage方法一测试,才发现在每次数据库读取都会增加内存。
看了db类找到原因了,原来db对象是放在全局里的,在数据库读取时会在他里面记录下执行的SQL。全局的程序没有执行完毕是不会释放的,这个脚本又是常驻内存执行的,解决方法也简单取消记录这个SQL就可以了。
经验:对于全局变量在循环调用里的内存使用要注意
<?php class db { public $sql; public function query($sql) { $this->sql[] = $sql; } } $db = new db(); while(1) { echo "m1:".memory_get_usage()." "; $db->query("select * from table"); echo "m2:".memory_get_usage()." "; usleep(10000); }
显然这不是内存泄漏的BUG,只是代码编码的问题。