[PHP]垃圾回收机制
PHP的垃圾回收机制:
2. 在zval结构体中定义了ref_count和is_ref,ref_count是引用计数,标识此zval被多少个变量引用,为0时会被销毁 3. 为了解决循环引用内存泄露问题,使用同步周期回收算法 顽固垃圾的产生过程 <?PHP $a = "new string"; ?> 代码中,$a变量内部存储信息为 a: (refcount_gc=1,is_ref_gc=0)='new string' 当把$a赋值给另外一个变量的时候,$a对应的zval的refcount_gc会加1 <?; $b = $a; ?> 此时$a和$b变量对应的内部存储信息为,$a和$b同时指向一个字符串"new string",它的refcount变成2 a,b: (refcount_gc=2,is_ref=0)='new string' 当用unset删除$b变量时,"new string" 的refcount_gc会减1变成1。 <?$a = "new string"; //a: (refcount_gc=1,is_ref_gc=0)='new string' $a; a,is_ref=0)='new string' unset($b); $a = array('meaning' => 'life','number' => 42); ?> $a内部存储信息为: a: (refcount=1,is_ref=0)=array ( 数组变量本身($a)在引擎内部实际上是一个哈希表,这张表中有两个zval项 meaning和number,所以实际上那一行代码中一共生成了3个zval,这3个zval都遵循变量的引用和计数原则,用图来表示: 下面在$a中添加一个元素,并将现有的一个元素的值赋给新的元素: <?); $a['name'] = $a['meaning']; ?> 那么$a的内部存储为,"life" 的ref_count变成2,42的ref_count是1: a: (refcount=1,is_ref=0)=array ( 如果将数组的引用赋值给数组中的一个元素,有意思的事情就会发生: <?array('one'$a[] = &; ?> 这样$a数组就有两个元素,一个索引为0,值为字符one,另外一个索引为1,为$a自身的引用,内部存储如下: a: (refcount=2,is_ref=1)=array ( array这个zval的ref_count是2,是一个环形引用 这时对$a进行unset,那么$a会从符号表中删除,同时$a指向的zval的refcount_gc减少1. <?PHP ); ; ); ?> 那么问题就产生了,$a已经不在符号表中,用户无法再访问此变量,但是$a之前指向的zval的refcount_gc变为1而不是0,因此不能被回收,从而产生内存泄露,新的GC要做的工作就是清理此类垃圾。 为了解决循环引用内存泄露问题,使用同步周期回收算法,这种ref_count减1后还大于0的会被作为疑似垃圾 (编辑:北几岛) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |