一个PHP脚本。一般都不用考虑内存泄漏和垃圾回收的问题,因为脚本很快就执行完退出了,每个页面处理结束,新建的simple_html_dom对象就会被销毁。
但当运行时间长,数据量大的时候,程序运行一段时间后,PHP脚本就占用了过多内存,然后报错:
PHP Fetal error:Allowed memory size of 1342117728 bytes exhausted
PHP垃圾回收机制
PHP5.3之前使用的垃圾回收机制是单纯的“引用计数”,也就是每个内存对象都分配了一个计数器,当内存对象被变量引用时,计数器+1,当变量引用撤掉时,计数器-1,当计数器等于0时,表明内存对象没有被使用,该内存对象则进行销毁,垃圾回收完成。
“引用计数”存在问题是当两个或多个对象互相引用形成环状后,内存对象的计数器不会消减为0,这一组内存对象没用时,不能进行回收,导致内存泄漏。
从PHP5.3开始,使用了新的垃圾回收机制,在引用计数的基础上,实现了复杂的算法,来检测内存对象中引用环的存在,以避免内存泄漏。
检查内存是否泄漏
可以简单的通过调用memory_get_usage函数查看内存使用情况来判断;memory_get_usage函数返回的内存使用数据据说不是很准确,可以使用xdebug扩展来获得更准确的内存使用情况。
int memory_get_usage ([ bool $real_usage = false ] )
//返回当前分配给你的PHP脚本的内存量,单位是字节(byte);
//real_usage如果设置为true,获取系统分配总的内存尺寸,包括未使用的页。如果未设置或者设置FALSE,仅仅报告实际使用内存量。
内存泄漏实例
class A{
private $b;
function __construct()
{
$this->b = new B($this);
}
function __destruct()
{
// TODO: Implement __destruct() method.
echo "A destruct";
}
}
class B {
private $a;
function __construct($a)
{
$this->a = $a;
}
function __destruct()
{
// TODO: Implement __destruct() method.
echo "B destruct";
}
}
for ($i = 0;;$i++) {
$a = new A();
echo memory_get_usage()."\n";
}