php 浅谈垃圾回收机制

Posted 平凡的胖子

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了php 浅谈垃圾回收机制相关的知识,希望对你有一定的参考价值。

php每创建一个变量,就会在zval中记录。一个zval变量容器,除了包含变量的类型和值,还包括两个字节的额外信息。

第一个是"is_ref",用来标识这个变量是否是属于引用集合(reference set), bool类型。

通过这个字节,php引擎才能把普通变量和引用变量区分开来,由于php允许用户通过使用&来使用自定义引用,zval变量容器中还有一个内部引用计数机制,来优化内存使用。

第二个额外字节是"refcount",用以表示指向这个zval变量容器的变量(也称符号即symbol)个数。所有的符号存在一个符号表中,其中每个符号都有作用域(scope),那些主脚本(比如:通过浏览器请求的的脚本)和每个函数或者方法也都有作用域。

安装Xdebug 来显示变量容器中的refcount 和 is_ref

demo1

<?php
$a = "hello world";
xdebug_debug_zval(‘a‘);
输出:
a:(refcount=1, is_ref=0)=‘hello world‘

新的变量a在当前作用域中生成了,同时生成了string类型 值为 ‘hello world’的变量容器。

将一个变量赋值给另外一个变量时会增加引用次数。

demo2

<?php
$a = "hello world";
$b = $a;
xdebug_debug_zval( ‘a‘ );
输出:
a: (refcount=2, is_ref=0)=‘hello world‘

输出发现refcount变成了2,同一个变量容器被$a和$b关联。在非必要时,php是不会复制已经生成的变量容器的,当refcount=0时,变量容器就会被销毁。

当任何关联到某个变量容器的变量离开它的作用域(函数执行结束)、或者对变量使用了unset()时,refcount就会-1

demo3

<?php
$a = "hello world";
$c = $b = $a;
xdebug_debug_zval( ‘a‘ );
unset( $b, $c );
xdebug_debug_zval( ‘a‘ );
输出:
a: (refcount=3, is_ref=0)=‘hello world‘
a: (refcount=1, is_ref=0)=‘hello world‘

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

复合类型:

demo4

<?php
$a = array( ‘name‘ => ‘life‘, ‘num‘ => 11 );
xdebug_debug_zval( ‘a‘ );
输出:
a: (refcount=1, is_ref=0)=array (
   ‘name‘ => (refcount=1, is_ref=0)=‘life‘,
   ‘num‘ => (refcount=1, is_ref=0)=11
)

这三个zval变量容器是:a,name,num

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

demo5

<?php
$a = array( ‘name‘ => ‘life‘, ‘num‘ => 11 );
$a[‘life‘] = $a[‘name‘];
xdebug_debug_zval( ‘a‘ );
输出:
a: (refcount=1, is_ref=0)=array (
   ‘name‘ => (refcount=2, is_ref=0)=‘life‘,
   ‘num‘ => (refcount=1, is_ref=0)=11,
   ‘life‘ => (refcount=2, is_ref=0)=‘life‘
)

发现name和life的refcount都变成了2,但是life是新添加的,说明值为‘life‘的zval变量容器其实是同一个,xdebug_debug_zval()并没有显示这个信息。删除其中一个元素,另一个的refcount也会减少,废话不多说,上demo

demo6

<?php
$a = array( ‘name‘ => ‘life‘, ‘num‘ => 11 );
$a[‘life‘] = $a[‘name‘];
unset( $a[‘name‘], $a[‘num‘] );
xdebug_debug_zval( ‘a‘ );
输出:
a: (refcount=1, is_ref=0)=array (
   ‘life‘ => (refcount=1, is_ref=0)=‘life‘
)

这样发现,一直都是refcount的值在改变,is_ref的值一直是false,当使用传引用的时is_ref的值才会发生变化。

其他的回收周期或者性能方面的考虑,有时间再不伤

 






以上是关于php 浅谈垃圾回收机制的主要内容,如果未能解决你的问题,请参考以下文章

浅谈Chrome V8引擎中的垃圾回收机制

浅谈垃圾回收机制

浅谈浏览器垃圾回收机制

PHP 垃圾回收机制

PHP垃圾回收机制理解

php回收机制