hi,你好!欢迎访问本站!登录
本站由简数采集腾讯云宝塔系统阿里云强势驱动
当前位置:首页 - PHP教程 - 正文 请牢记本站网址www.sosophp.cn

经由过程实例细致解说PHP垃圾接纳机制【php教程】

2019-12-01PHP教程搜搜PHP网47°c
A+ A-
PHP垃圾接纳机制

1. PHP能够自动举行内存治理,消灭不需要的对象,重要运用了援用计数

2. 在zval构造体中定义了ref_count和is_ref , ref_count是援用计数 ,标识此zval被多少个变量援用 , 为0时会被烧毁。is_ref标识是不是运用的 &取地点符强迫援用

3. 为了处理轮回援用内存泄漏题目 , 运用同步周期接纳算法。

比方当数组或对象轮回的援用自身 , unset掉数组的时刻 , 当refcount-1后还大于0的 , 就会被当做疑似垃圾 , 会举行遍历 ,而且模仿的删除一次refcount-1假如是0就删除 ,假如不是0就恢复固执垃圾的发作历程:

<?php
    $a = "new string";
?>
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变成2a,b: (refcount_gc=2,is_ref=0)='new string'

当用unset删除$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: (refcount_gc=2, is_ref=0)='new string'
    unset($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: (refcount=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举行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就恢复。

想相识更多相干内容请接见ki4网:PHP视频教程

以上就是经由过程实例细致解说PHP垃圾接纳机制的细致内容,更多请关注ki4网别的相干文章!

  选择打赏方式
微信赞助

打赏

QQ钱包

打赏

支付宝赞助

打赏

  选择分享方式
  移步手机端
经由过程实例细致解说PHP垃圾接纳机制【php教程】

1、打开你手机的二维码扫描APP
2、扫描左则的二维码
3、点击扫描获得的网址
4、可以在手机端阅读此文章