`

PHP垃圾回收机制-引用计数基本知识

    博客分类:
  • PHP
 
阅读更多

每个php变量存在一个叫"zval"的变量容器中。一个zval变量容器,除了包含变量的类型和值,还包括两个字节的额外信息。第一个是"is_ref",是个bool值,用来标识这个变量是否是属于引用集合(reference set)。通过这个字节,php引擎才能把普通变量和引用变量区分开来,由于php允许用户通过使用&来使用自定义引用,zval变量容器中还有一个内部引用计数机制,来优化内存使用。第二个额外字节是"refcount",用以表示指向这个zval变量容器的变量(也称符号即symbol)个数。所有的符号存在一个符号表中,其中每个符号都有作用域(scope),那些主脚本(比如:通过浏览器请求的的脚本)和每个函数或者方法也都有作用域。

当一个变量被赋常量值时,就会生成一个zval变量容器,如下例这样:

例1:

 

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

 在上例中,新的变量a,是在当前作用域中生成的。并且生成了类型为 string 和值为new string的变量容器。在额外的两个字节信息中,"is_ref"被默认设置为 FALSE,因为没有任何自定义的引用生成。"refcount" 被设定为 1,因为这里只有一个变量使用这个变量容器. 注意到当"refcount"的值是1时,"is_ref"的值总是FALSE. 如果你已经安装了» Xdebug,你能通过调用函数 xdebug_debug_zval()显示"refcount"和"is_ref"的值。

 

例2:

 

<?php
xdebug_debug_zval('a');
?>

 以上例程会输出:

 

 

a:
(refcount=1, is_ref=0),string 'new string' (length=10)

 

 

把一个变量赋值给另一变量将增加引用次数(refcount).

例3:

 

<?php
$a = "new string";
$b = $a;
xdebug_debug_zval( 'a' );
?>

 以上例程会输出:

 

 

a:
(refcount=2, is_ref=0),string 'new string' (length=10)

 这时,引用次数是2,因为同一个变量容器被变量 a 和变量 b关联.当没必要时,php不会去复制已生成的变量容器。变量容器在”refcount“变成0时就被销毁. 当任何关联到某个变量容器的变量离开它的作用域(比如:函数执行结束),或者对变量调用了函数 unset()时,”refcount“就会减1,下面的例子就能说明:

 

 

<?php
$a = "new string";
$c = $b = $a;
xdebug_debug_zval( 'a' );
unset( $b, $c );
xdebug_debug_zval( 'a' );
?>

 以上例程会输出:

 

 

a:
(refcount=3, is_ref=0),string 'new string' (length=10)
a:
(refcount=1, is_ref=0),string 'new string' (length=10)

 如果我们现在执行 unset($a);,包含类型和值的这个变量容器就会从内存中删除。

 

-----------------------------------

 

复合类型(Compound Types)

当考虑像 array和object这样的复合类型时,事情就稍微有点复杂. 与 标量(scalar)类型的值不同,array和 object类型的变量把它们的成员或属性存在自己的符号表中。这意味着下面的例子将生成三个zval变量容器。

例5:

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

 以上例程的输出类似于:

a:
(refcount=1, is_ref=0),
array (size=2)
  'meaning' => (refcount=1, is_ref=0),string 'life' (length=4)
  'number' => (refcount=1, is_ref=0),int 42

 或者图表形式:

 

这三个zval变量容器是: ameaning和 number。增加和减少”refcount”的规则和上面提到的一样. 下面, 我们在数组中再添加一个元素,并且把它的值设为数组中已存在元素的值:

例6:

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

 以上例程的输出类似于:

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

 或者图表形式:



 

从以上的xdebug输出信息,我们看到原有的数组元素和新添加的数组元素关联到同一个"refcount"2的zval变量容器. 尽管 Xdebug的输出显示两个值为'life'的 zval 变量容器,其实是同一个。 函数xdebug_debug_zval()不显示这个信息,但是你能通过显示内存指针信息来看到。

删除数组中的一个元素,就是类似于从作用域中删除一个变量. 删除后,数组中的这个元素所在的容器的“refcount”值减少,同样,当“refcount”为0时,这个变量容器就从内存中被删除,下面又一个例子可以说明:

例7:

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

 以上例程的输出类似于:

a:
(refcount=1, is_ref=0),
array (size=1)
  'life' => (refcount=1, is_ref=0),string 'life' (length=4)

 

现在,当我们添加一个数组本身作为这个数组的元素时,事情就变得有趣,下个例子将说明这个。例中我们加入了引用操作符,否则php将生成一个复制。

例8:

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

 以上例程的输出类似于:

a:
(refcount=2, is_ref=1),
array (size=2)
  0 => (refcount=1, is_ref=0),string 'one' (length=3)
  1 => (refcount=2, is_ref=1),
    &array

 或者图表形式:



 能看到数组变量 (a) 同时也是这个数组的第二个元素(1) 指向的变量容器中“refcount”为 2。上面的输出结果中的"..."说明发生了递归操作, 显然在这种情况下意味着"..."指向原始数组(官方文档是这么写的,但是我的执行环境是上面那样&array)。

跟刚刚一样,对一个变量调用unset,将删除这个符号,且它指向的变量容器中的引用次数也减1。所以,如果我们在执行完上面的代码后,对变量$a调用unset, 那么变量 $a 和数组元素 "1" 所指向的变量容器的引用次数减1, 从"2"变成"1". 下例可以说明:

例9:

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

 官方的结果:

a:
(refcount=1, is_ref=1),
array (size=2)
  0 => (refcount=1, is_ref=0),string 'one' (length=3)
  1 => (refcount=2, is_ref=1),
    &array

 

但是在自己环境下执行结果是空的,说明unset($a)后refcount = 0了;这一点还没理解。

----------------------------------

清理变量容器的问题(Cleanup Problems)

尽管不再有某个作用域中的任何符号指向这个结构(就是变量容器),由于数组元素“1”仍然指向数组本身,所以这个容器不能被清除 。因为没有另外的符号指向它,用户没有办法清除这个结构,结果就会导致内存泄漏。庆幸的是,php将在请求结束时清除这个数据结构,但是在php清除之前,将耗费不少空间的内存。如果你要实现分析算法,或者要做其他像一个子元素指向它的父元素这样的事情,这种情况就会经常发生。当然,同样的情况也会发生在对象上,实际上对象更有可能出现这种情况,因为对象总是隐式的被引用。

如果上面的情况发生仅仅一两次倒没什么,但是如果出现几千次,甚至几十万次的内存泄漏,这显然是个大问题。在长时间运行的脚本,比如请求基本上不会结束的守护进程(deamons)或者单元测试中的大的套件(sets)中,在给 eZ 组件库的模板组件做单元测试时,后者(指单元测试中的大的套件)就会出现问题.它将需要耗用2GB的内存,而一般的测试服务器没有这么大的内存空间。

参考:http://php.net/manual/zh/features.gc.refcounting-basics.php

 

  • 大小: 9.1 KB
  • 大小: 10 KB
  • 大小: 5.8 KB
分享到:
评论

相关推荐

    掌握PHP垃圾回收机制详解

    本文主要和大家分享掌握php垃圾回收机制的知识,希望能帮助到大家。 引用计数基本知识 官网的解答如下 每个php变量存在一个叫”zval”的变量容器中一个zval变量容器,除了包含变量的类型和值 ,还包括两个字节的额外...

    解读PHP中的垃圾回收机制

    PHP内存管理和垃圾回收机制的知识点还包括使用PHP内置函数memory_get_usage()来监控内存使用情况,以及使用__destruct()魔术方法来实现资源的自动清理。__destruct()魔术方法是在垃圾对象被回收时执行的析构函数,...

    深入理解PHP内核-php宝典

    - **垃圾回收机制**:除了引用计数之外,PHP还引入了一种循环检测算法来处理难以检测的内存泄露情况,比如循环引用。 #### 四、结语 通过上述分析,《深入理解PHP内核—PHP宝典》不仅详细介绍了PHP内核中的关键概念...

    PHP程序设计-3期(KC016) 拓展知识2-3 自定义变量.doc

    2. `zend_uint refcount__gc`: 变量的引用计数,用于垃圾回收机制,当一个变量不再被任何变量引用时,它的内存会被释放。 3. `zend_uchar type`: 表示变量的当前数据类型,PHP支持的8种基本数据类型(整型、浮点型、...

    PHP内核介绍及扩展开发指南.pdf

    当`refcount`降到0时,表示没有变量再使用这个`zval`,因此可以被垃圾回收机制回收。`is_ref`成员指示该`zval`是否被引用赋值过。 #### zval状态切换 在PHP中,变量可以是引用类型或非引用类型。非引用类型的变量...

    开源项目-walu-phpbook.zip

    - 引用计数与垃圾回收:理解PHP如何跟踪和释放不再使用的内存。 - 类与对象的实现:探讨PHP的面向对象特性,如类、对象、继承、接口、抽象类等的内部实现。 - 错误与异常处理:深入理解PHP的错误报告机制和异常...

    PHP基础教程PHP简介

    - **PHP4.0**:2000年5月22日发布,引入了许多新特性,如引用计数垃圾回收机制等。 - **PHP5**:2004年7月发布,引入面向对象的支持,并且对许多内部结构进行了优化。 - **普及程度**:截至2023年8月,全球已有...

    php_extension_writing

    如果`refcount`为0,则意味着没有引用指向该`zval`,这通常是垃圾回收的信号。 - `is_ref`字段用于标记当前`zval`是否为引用。如果是引用,则`is_ref`的值为1;如果不是引用,则值为0。 - 复制时写入(Copy-On-...

    变量在 PHP7 内部的实现(二)

    zend_refcounted结构体包含了引用计数、值类型信息以及垃圾回收的相关信息gc_info和类型标志位flags。 字符串类型在PHP7中的实现是通过zend_string结构体来完成的。这个结构体包含了引用计数的头,一个哈希缓存...

    深入理解PHP内核.mobi

    这通常涉及到 Zend Engine 的垃圾回收机制,如引用计数法和标记清除算法,以及内存池的概念,这些都是影响程序性能的关键因素。 4. **Zend虚拟机(Zend VM)**:作为PHP的执行引擎,Zend VM是如何工作的?它如何将PHP...

    PHP内核介绍及扩展开发指南

    当引用计数变为0时,`zval`将被垃圾回收器释放。 ##### 1.1.3 `zval`状态 `zval`中的`is_ref`字段表示当前`zval`是否是引用类型。对于非引用型变量,如果多个变量引用同一个`zval`,则修改其中一个变量时,其他...

    php面向对象手册

    #### 八、PHP 的垃圾回收机制 PHP 使用引用计数机制来进行内存管理。当一个对象不再被任何变量引用时,它的内存会被自动释放。此外,PHP 还提供了手动清理资源的方法,例如在析构函数中关闭数据库连接等。 #### 九...

    深入理解PHP内核,PHP扩展开发

    PHP的内存管理策略包括引用计数和垃圾回收机制。理解这些机制对于防止内存泄漏和提高内存使用效率至关重要。开发者需要了解如何正确管理资源,如对象和数组,以减少内存开销。 4. **C语言扩展PHP功能** PHP的扩展...

    Programming in Objective-C 5th Edition

    - **引用计数:** Objective-C使用自动引用计数(Automatic Reference Counting, ARC)机制来自动管理内存。 - **垃圾回收:** 对于不需要的对象自动进行回收处理,减轻开发者负担。 - **弱引用与强引用:** 掌握这两...

    PHP面试题集锦

    8. **PHP的垃圾回收机制(Garbage Collection, GC)是如何工作的?** PHP的GC自动回收不再使用的对象所占的内存。当一个对象没有引用指向它时,该对象成为垃圾。PHP通过引用计数和周期回收策略来判断和清理垃圾对象...

    php开发高级技巧结合案例

    2. PHP的垃圾回收机制:了解何时以及如何释放不再使用的内存。 3. Zend Engine:PHP的执行引擎,深入学习其工作原理,可以帮助优化代码执行效率。 4. PHP扩展开发:了解如何编写C扩展,以提升特定功能的性能或实现...

    PHP赋值的内部是如何跑的详解

    当对象的引用计数降到0时,对象才会被垃圾回收机制回收。 通过以上的讨论,我们可以了解到,PHP赋值的内部机制涉及到变量存储的结构、引用计数、写时复制机制以及对象引用计数等概念。掌握这些知识有助于我们编写...

    2021-2022计算机二级等级考试试题及答案No.10073.docx

    当一个对象不再被任何变量引用时,它可能会被垃圾回收机制回收。在选项B中,`u`指向了`v`,导致原来的`u`对象失去引用,因此可能会被回收。故正确答案为B。 ### 6. 数据结构 - **知识点**:数据结构分为线性和非...

Global site tag (gtag.js) - Google Analytics