资讯专栏INFORMATION COLUMN

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

dreambei / 2509人阅读

摘要:后者的例子在给巨大的一个知名的组件库的模板组件做单元测试时,就可能会出现问题。

PHP的垃圾回收机制

垃圾回收器,全称Garbage Collection,简称GC,5.3版本之前 只是简单判断变量的zval的refcount是否为0,是的话就释放,不是直至进程结束(隐藏着变量内存溢出的风险).

如果你已经安装了» Xdebug,你能通过调用函数 xdebug_debug_zval()显示"refcount"和"is_ref"的值。

引用计数基本知识 1、一个zval变量容器:
内容
类型
is_ref 是个bool值,用来标识这个变量是否是属于引用集合(reference set) zval变量容器中还有一个内部引用计数机制,来优化内存使用
refcount 指向这个zval变量容器的变量(也称符号即symbol)个数 所有的符号存在一个符号表中,其中每个符号都有作用域(scope),那些主脚本(比如:通过浏览器请求的的脚本)和每个函数或者方法也都有作用域。

当一个变量被赋常量值时,就会生成一个zval变量容器

1、 生成一个新的zval容器

参考上面的定义,分别是:

生成的变量容器,其中:

类型 string

值 new string

is_ref false

Refcount 1

2、增加一个zval的引用计数

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

类型 string

值 new string

is_ref true

Refcount 2

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

3、 减少引用计数


a: (refcount=3, is_ref=0)="new string"
a: (refcount=1, is_ref=0)="new string"

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

4、复合类型(Compound Types) ¶

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

 "life", "number" => 42 );
xdebug_debug_zval( "a" );
?>

输出

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

看图说话

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

添加一个已经存在的元素到数组中

 "life", "number" => 42 );
$a["life"] = $a["meaning"];
xdebug_debug_zval( "a" );
?>

输出

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

有图有真相

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

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

 "life", "number" => 42 );
$a["life"] = $a["meaning"];
unset( $a["meaning"], $a["number"] );
xdebug_debug_zval( "a" );
?>

输出

a: (refcount=1, is_ref=0)=array (
   "life" => (refcount=1, is_ref=0)="life"
)

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

把数组作为一个元素添加到自己

输出

a: (refcount=2, is_ref=1)=array (
   0 => (refcount=1, is_ref=0)="one",
   1 => (refcount=2, is_ref=1)=...
)

看图说话

能看到数组变量 (a) 同时也是这个数组的第二个元素(1) 指向的变量容器中“refcount”为 2。上面的输出结果中的"..."说明发生了递归操作, 显然在这种情况下意味着"..."指向原始数组。

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


输出

(refcount=1, is_ref=1)=array (
   0 => (refcount=1, is_ref=0)="one",
   1 => (refcount=1, is_ref=1)=...
)

看图说话

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

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

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

PHP手册

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/31004.html

相关文章

  • PHPer面试指南-PHP

    摘要:本书的地址篇收集了一些常见的基础进阶面试题,基础的面试题不再作答。如何实现持久化持久化,将在内存中的的状态保存到硬盘中,相当于备份数据库状态。相当于备份数据库接收到的命令,所有被写入的命令都是以的协议格式来保存的。 本书的 GitHub 地址:https://github.com/todayqq/PH... PHP 篇收集了一些常见的基础、进阶面试题,基础的面试题不再作答。 基础篇 ...

    stackvoid 评论0 收藏0
  • 【转】浅谈PHP5中垃圾回收算法(Garbage Collection)演化

    摘要:所有这些类型,在内部统一用一个叫做的结构表示,在源代码中这个结构名称为。的具体定义在源代码的文件中,下面是相关代码的摘录。 【转】浅谈PHP5中垃圾回收算法(Garbage Collection)的演化 前言 PHP是一门托管型语言,在PHP编程中程序员不需要手工处理内存资源的分配与释放(使用C编写PHP或Zend扩展除外),这就意味着PHP本身实现了垃圾回收机制(Garbage C...

    AdolphLWQ 评论0 收藏0
  • PHP回收周期

    摘要:原文回收周期,增加一篇论文在底部。就是说,仅仅在引用计数减少到非零值时,才会产生垃圾周期。如果他们是循环引用周期的一部分,将永不能被清除进而导致内存泄漏。这个函数将返回使用这个算法回收的周期数。引用计数系统中的同步周期回收 原文:回收周期(Collecting Cycles) ,增加一篇论文在底部。 以下过程仅对数组和对象类型起作用。 传统上,像以前的 php 用到的引用计数内存机制,...

    Astrian 评论0 收藏0
  • 能让你更早下班Python垃圾回收机制

    摘要:内部通过引用计数机制来统计一个对象被引用的次数。下一步,就该被我们的垃圾回收器给收走了。而我们垃圾回收机制只有当引用计数为的时候才会释放对象。以空间换时间的方法提高垃圾回收效率。 人生苦短,只谈风月,谈什么垃圾回收。据说上图是某语言的垃圾回收机制。。。我们写过C语言、C++的朋友都知道,我们的C语言是没有垃圾回...

    pumpkin9 评论0 收藏0
  • PHP垃圾回收机制

    摘要:在中算法,当节点缓冲区满了之后,垃圾分析算法就会启动,并且会释放掉发现的垃圾,从而回收内存。在编程中程序员不需要手动处理内存资源分配与释放,意味着本身实现了垃圾回收处理机制。 PHP是一种弱类型的脚本语言,弱类型不表示PHP变量没有类型的区别,PHP变量有8种原始类型:四种标量类型: boolean(布尔值) integer(整型) float(浮点型) 两种复合类型: arra...

    luck 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<