$arr = array( 'a' => 'aaa', 'b' => "bbb" ); xdebug_debug_zval( 'arr' ); $arr['aaa'] = $arr['a']; xdebug_debug_zval( 'arr' ); ?> 结果: arr: (refcount=1, is_ref=0), array 'a' => (refcount=1, is_ref=0),string 'aaa' (length=3) 'b' => (refcount=1, is_ref=0),string 'bbb' (length=3) arr: (refcount=1, is_ref=0), array 'a' => (refcount=2, is_ref=0),string 'aaa' (length=3) 'b' => (refcount=1, is_ref=0),string 'bbb' (length=3) 'aaa' => (refcount=2, is_ref=0),string 'aaa' (length=3) 可以看到看到原有的数组元素和新添加的数组元素关联到同一个"refcount"2的zval变量容器.这里我也只是起到抛砖引玉的作用。 上面我们只是简单的使用了unset,null,mysql_close,__destruct,xdebug_debug_zval 接着往下看 查看内存是否泄露 看是否有该释放的内存没有被释放,可以简单的通过 调用 memory_get_usage 函数查看内存使用情况来判断;memory_get_usage 函数返回的内存使用数据据说不是很准确,可以使用 php 的 xdebug 扩展来获得更准确翔实的内存使用情况。
代码如下 | 复制代码 | class A{ private $b; function __construct(){ $this->b = new B($this); } function __destruct(){ //echo "A destructn"; } } class B{ private $a; function __construct($a){ $this->a = $a; } function __destruct(){ //echo "B descturctn"; } } for($i=0;;$i ){ $a = new A(); if($i00 == 0){ echo memory_get_usage()."n"; } } } 上面就构造了一个会产生环状引用的例子;每次创建一个A对象的实例a,a就创建一个B对象的实例b,同时让b引用a ;这样,每个A对象永远被一个B引用,而每个B对象同时被一个对象A引用;引用环就这样产生了。 在php5.2的环境下执行这段代码,会发现内存使用在单调上涨,也没有A和B的析构函数被执行后输出的“A/B desctruct”信息;直到内存耗尽,输出“PHP Fatal error: Allowed memory size of 134217728 bytes exhausted (tried to allocate 40 bytes)”。 在php5.3的环境下执行这段代码,则发现内存使用在上跳下窜,但是永远没有超过一个限额;程序也会输出大量的“A/B desctruct”,这说明析构函数被调用了。 我的同事的程序中,就存在这种引用的环路,而他的脚本,实在php5.2.3下执行的。simple_html_dom工具中,有两个类,分别是simple_html_dom和simple_html_dom_node,前者中有一个数组成员变量nodes,数组中每个元素都是一个simple_html_dom_node对象;而每个simple_html_dom_node对象都有一个成员变量dom,该dom的值就是前面的simple_html_dom对象——这样就形成了一个漂亮的引用环,导致了内存泄露。解决的办法也很简单,就是simple_html_dom对象在使用完毕时,主动调用其clear函数,清空其成员变量nodes,环就被打破了,内存泄露也就不会发生了。 3.其他: 1)垃圾回收的时机 Php中,引用计数为0,则内存立刻释放;也就是说,不存在环状引用的变量,离开变量的作用域,内存被立刻释放。 环状引用检测则是在满足一定条件下触发,所以在上面的例子中,会看到使用的内存有大幅度的波动;也可以通过 gc_collect_cycles 函数来主动进行环状引用检测。 2) &符号的影响 显式引用一个变量,会增加该内存的引用计数: $a = "something"; $b = &$a; 此时unset($a), 但是仍有$b指向该内存区域的引用,内存不会释放。 3)unset函数的影响 unset只是断开一个变量到一块内存区域的连接,同时将该内存区域的引用计数-1;在上面的例子中,循环体内部,$a=new A(); unset($a);并不会将$a的引用计数减到零; 4)= null 操作的影响; $a = null 是直接将$a 指向的数据结构置空,同时将其引用计数归0。 5)脚本执行结束的影响 脚本执行结束,该脚本中使用的所有内存都会被释放,不论是否有引用环。 |
|