首页 > 作文

[PHP]垃圾回收机制

更新时间:2023-04-07 07:27:29 阅读: 评论:0

php的垃圾回收机制:

1. php可以自动进行内存管理,清除不需要的对象,主要使用了引用计数

2. 在zval结构体中定义了ref_count和is_ref , ref_count是引用计数 ,标识此zval被多少个变量引用 , 为0时会被销毁
is_ref标识是否使用的 &取地址符强制引用

3. 为了解决循环引用内存泄露问题 , 使用同步周期回收算法
比如当数组或对象循环的引用自身 , unt掉数组的时候 , 当refcount-1后还大于0的 , 就会被当成疑似垃圾 , 会进行遍历 ,并且模拟的删除一次refcount-1如果是0就删除 ,如果不是0就恢复

顽固垃圾的产生过程

<?php    $a = "new string";?>

代码中,$a变量内部存储信息为

a: (refcount_gc=1, is_ref_gc=0)=’new string’

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

<?php    $a = "new string";    $b = $a;?>

此时$a和$b变量对应的内部存储信息为,$a和$b同时指向一个字符串”new string” ,它的refcount变成2

a,b: (refcount_gc=2, is_ref=0)=’new string’

当用unt删除$b变量时,”new string” 的refcount_gc会减1变成1。

<?php    $a = "new string"; //a: (refcount_gc=1, is_ref_gc=0)='new string'    $b = $a;           //a,b: (refcoun英雄联盟轮子妈t_gc=2, is_ref=0)='new string'    unt($b);         //a: (refcount_gc=1, is_ref=0)='new string'?>

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

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

$a内部存储信息为:

a: (refcount=1, is_ref=0)=array (
‘meaning’ => (refcount=1, is_ref=0)=’life’,
‘number’ => (refcount=1, is_ref=0)=42
)

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

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

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

那么$a的内部存储为 , “life” 的ref_count变成2 , 42的ref_count是1形容颜色:

a: (re一件烦心事作文fcount=1, is_ref=0)=array (
‘meaning’ => (refcount=2, is_ref=0)=’life’,
‘number’ => (refcount=1, is_ref=0)=42,
‘name’ => (refcount=2, is_ref=0)=’life’
)

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

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

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

a: (refcount=2, is_ref=1)=array (
0 => (refcount=1, is_ref=0)=’one’,
1 => (refcount=2, is_ref=1)=…
)

array这个zval的ref_count是2 , 是一个环形引用

这时对$a进行unt,那么$a会从符号表中删除,同时$a指向的zval的refcount_gc减少1.

<?php$a = array('one');$a[] = &$a;unt($a);?>

那么问题就产生了,$a已经不在符号表中,用户无法再访问此变量,但是$a之前指向的z送孟浩然之广陵val的refcount_gc变为1而不是0,因此不能被回收,从而产生内存泄露,新的gc要做的工作就是清理此类垃圾。

为了解决循环引用内存泄露问题 , 使用同步周期回收算法 , 这种ref_count减1后还大于0的会被作为疑似垃圾
比如当数组或对象循环的引用自身 , unt掉数组的时候 , 当refcount-1后给予的反义词还大于0的 , 会进行遍历 ,并且模拟的删除一次refcount-1如果是0就删除 ,如果不是0就恢复

本文发布于:2023-04-07 07:27:27,感谢您对本站的认可!

本文链接:https://www.wtabcd.cn/fanwen/zuowen/534c079deab4f62dd6684960d8cd70fc.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

本文word下载地址:[PHP]垃圾回收机制.doc

本文 PDF 下载地址:[PHP]垃圾回收机制.pdf

标签:变量   数组   元素   垃圾
相关文章
留言与评论(共有 0 条评论)
   
验证码:
Copyright ©2019-2022 Comsenz Inc.Powered by © 专利检索| 网站地图