ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

通过实例详细讲解PHP垃圾回收机制

2019-09-01 16:02:19  阅读:255  来源: 互联网

标签:实例 refcount zval gc 引用 数组 讲解 PHP ref


PHP垃圾回收机制:
1. PHP可以自动进行内存管理,清除不需要的对象,主要使用了引用计数
2. 在zval结构体中定义了ref_count和is_ref , ref_count是引用计数 ,标识此zval被多少个变量引用 , 为0时会被销毁。is_ref标识是否使用的 &取地址符强制引用
3. 为了解决循环引用内存泄露问题 , 使用同步周期回收算法。
比如当数组或对象循环的引用自身 , unset掉数组的时候 , 当refcount-1后还大于0的 , 就会被当成疑似垃圾 , 会进行遍历 ,并且模拟的删除一次refcount-1如果是0就删除 ,如果不是0就恢复顽固垃圾的产生过程:

1 <?php
2     $a = "new string";
3 ?>
1 a: (refcount_gc=1, is_ref_gc=0)='new string'

当把$a赋值给另外一个变量的时候,$a对应的zval的refcount_gc会加1

1 <?php
2     $a = "new string";
3     $b = $a;
4 ?>

此时$a和$b变量对应的内部存储信息为,$a和$b同时指向一个字符串"new string" ,它的refcount变成2a,b: (refcount_gc=2,is_ref=0)='new string'
当用unset删除$b变量时,"new string" 的refcount_gc会减1变成1

1 <?php
2     $a = "new string"; //a: (refcount_gc=1, is_ref_gc=0)='new string'
3     $b = $a;           //a,b: (refcount_gc=2, is_ref=0)='new string'
4     unset($b);         //a: (refcount_gc=1, is_ref=0)='new string'
5 ?>

对于普通的变量来说,这一切很正常,但是在复合类型变量(数组和对象)中,会发生比较有意思的事情:

1 <?php
2     $a = array('meaning' => 'life', 'number' => 42);
3 ?>

$a内部存储信息为:

1 a: (refcount=1, is_ref=0)=array (
2 'meaning' => (refcount=1, is_ref=0)='life',
3 'number' => (refcount=1, is_ref=0)=42
4 )

数组变量本身($a)在引擎内部实际上是一个哈希表,这张表中有两个zval项 meaning和number,所以实际上那一行代码中一共生成了3个zval,这3个zval都遵循变量的引用和计数原则,用图来表示:

下面在$a中添加一个元素,并将现有的一个元素的值赋给新的元素:

1 <?php
2     $a = array('meaning' => 'life', 'number' => 42);
3     $a['name'] = $a['meaning'];
4  ?>

那么$a的内部存储为 , "life" 的ref_count变成2 , 42的ref_count是1:

1 a: (refcount=1, is_ref=0)=array (
2 'meaning' => (refcount=2, is_ref=0)='life',
3 'number' => (refcount=1, is_ref=0)=42,
4 'name' => (refcount=2, is_ref=0)='life'
5 )

如果将数组的引用赋值给数组中的一个元素,有意思的事情就会发生:

1 <?php
2     $a = array('one');
3     $a[] = &$a;
4 ?>

这样$a数组就有两个元素,一个索引为0,值为字符one,另外一个索引为1,为$a自身的引用,内部存储如下:

1 a: (refcount=2, is_ref=1)=array (
2 0 => (refcount=1, is_ref=0)='one',
3 1 => (refcount=2, is_ref=1)=…
4 )

array这个zval的ref_count是2 , 是一个环形引用。
这时对$a进行unset,那么$a会从符号表中删除,同时$a指向的zval的refcount_gc减少1.
那么问题就产生了,$a已经不在符号表中,用户无法再访问此变量,但是$a之前指向的zval的refcount_gc变为1而不是0,因此不能被回收,从而产生内存泄露,新的GC要做的工作就是清理此类垃圾。
为了解决循环引用内存泄露问题 , 使用同步周期回收算法 , 这种ref_count减1后还大于0的会被作为疑似垃圾。
比如当数组或对象循环的引用自身 , unset掉数组的时候 , 当refcount-1后还大于0的 , 会进行遍历 ,并且模拟的删除一次refcount-1如果是0就删除 ,如果不是0就恢复。

 

链接:https://mp.weixin.qq.com/s/oI80p-fV5RwYH0s6UkVudg

标签:实例,refcount,zval,gc,引用,数组,讲解,PHP,ref
来源: https://www.cnblogs.com/clubs/p/11442554.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有