php引用计数与变量引用
每个php5.5变量都存储在一个叫做zval的变量容器中。
一个zval变量容器,除了包含变量的类型与值外,还包含两个字节的额外信息:
1、第一个是“is_ref”,是个bool型,用来标识这个变量是否属于引用集合(reference set),若属于则其值为1,否则为0。
有个这个变量php引擎就能够将普通变量与引用变量区分开来。
2、第二个是“refcount”,用来表示指向这个zval变量(符号)的个数。每个符号都有作用域(scope),那些主脚本和函数或者方法也都有作用域。
所有的符号都存在一个符号表中。
当一个变量被赋值一个常量值时,就会生成一个zval变量容器,如下例:
<?php
$a = "Hello world";
?>
这个时候执行以下程序得到$a变量指向zval容器中的is_ref与refcount值
<?php
$a = "Hello world";
print_r(xdebug_debug_zval('a'));
?>
a: (refcount=1, is_ref=0)='Hello world'
下面,我们进行如下实验,来探讨引用赋值与普通赋值。
首先,使$b指向$a,查看is_ref、 refcount,如下:
<?php
$a = "Hello world";
$b = $a;
print_r(xdebug_debug_zval('a'));
print_r(xdebug_debug_zval('b'));
?>
a: (refcount=2, is_ref=0)='Hello world'
b: (refcount=2, is_ref=0)='Hello world'
让$b引用$a,查看is_ref refcount,如下
<?php
$a = "Hello world";
$b = &$a;
print_r(xdebug_debug_zval('a'));
print_r(xdebug_debug_zval('b'));
?>
a: (refcount=2, is_ref=1)='Hello world'
b: (refcount=2, is_ref=1)='Hello world'
从上我们可以分析出,当有变量引用相应zval容器时,is_ref为1。
我们进一步分析,我们把$b 引用$a,$c指向$a,如下
<?php $a = "Hello world"; $b = &$a; $c = $a; print_r(xdebug_debug_zval('a')); print_r(xdebug_debug_zval('b')); print_r(xdebug_debug_zval('c')); ?>
打印结果如下
a: (refcount=2, is_ref=1)='Hello world'
b: (refcount=2, is_ref=1)='Hello world'
c: (refcount=1, is_ref=0)='Hello world'
可见,这个时候php5.5引擎为$c重新建立了一个zval容器,容器中的数据类型、值与$a指向的容器中的完全相同,不同的是其refcount与is_ref的值。
因此,我们可以看出,php5.5的zval容器中的is_ref变量要么标识引用集合,要么标识普通集合,当两者都有时,他将克隆zval容器,来解决冲突问题。
总结:
1、在php5.5以后,“变量赋值”都是指向赋值,即将某个变量指向特定的zval容器。
2、“变量引用”则是将变量与变量进行绑定,若绑定的变量中有一个变量改变了指向,则相互绑定的其他变量的指向也随着改变。
若变量重新引用变量,则其原来的变量绑定解除,转而绑定新的变量。如下代码:
<?php function foo(&$var) { $var =& $GLOBALS["baz"]; } foo($bar); ?>
这将使 foo 函数中的 $var 变量在函数调用时和 $bar 绑定在一起,但接着又被重新绑定到了 $GLOBALS["baz"] 上面。不可能通过引用机制将 $bar 在函数调用范围内绑定到别的变量上面,因为在函数 foo 中并没有变量$bar(它被表示为 $var,但是 $var 只有变量内容而没有调用符号表中的名字到值的绑定)。可以使用引用返回来引用被函数选择的变量。
哈尔滨品用软件有限公司致力于为哈尔滨的中小企业制作大气、美观的优秀网站,并且能够搭建符合百度排名规范的网站基底,使您的网站无需额外费用,即可稳步提升排名至首页。欢迎体验最佳的哈尔滨网站建设。