之变量分离/引用(Scope in PHP)


在前面的文章中我已经介绍了PHP的变量的内部表示(深入理解PHP原理之变量(Variables inside PHP)),以及PHP中作用域的实现机制(深入理解PHP原理之变量作用域(Scope inside PHP))。这节我们就接着前面的文章,继续介绍PHP中变量分离和引用的概念:

首先我们回顾一下zval的结构:

1.       struct _zval_struct {

2.               /* Variable information */

3.               zvalue_value value;             /* value */

4.               zend_uint refcount;

5.               zend_uchar type;        /* active type */

6.               zend_uchar is_ref;

7.       };

其中的refcountis_ref字段我们一直都没有介绍过,我们知道PHP是一个长时间运行的服务器端的脚本解释器。那么对于它来说,效率和资源占用率是一个很重要的衡量标准,也就是说,PHP必须尽量介绍内存占用率,考虑下面这段代码:

1.       <?php

2.          $var = "laruence";

3.          $var_dup = $var;

4.          unset($var);

5.       ?>

第一行代码创建了一个字符串变量,申请了一个大小为9字节的内存,保存了字符串”laruence”和一个NULL(\0)的结尾。
第二行定义了一个新的字符串变量,并将变量var的值复制给这个新的变量。
第三行unset了变量var

这样的代码在我们平时的脚本中是很常见的,如果PHP对于每一个变量赋值都重新分配内存,copy数据的话,那么上面的这段代码公要申请18个字节的内存空间,而我们也很容易的看出来,上面的代码其实根本没有必要申请俩份空间,呵呵,PHP的开发者也看出来了:

我们之前讲过,PHP中的变量是用一个存储在symbol_table中的符号名,对应一个zval来实现的,比如对于上面的第一行代码,会在symbol_table中存储一个值”var”, 对应的有一个指针指向一个zval结构,变量值”laruence”保存在这个zval中,所以不难想象,对于上面的代码来说,我们完全可以让”var””var_dup”对应的指针都指向同一个zval就可以了。

PHP也是这样做的,这个时候就需要介绍我们之前一直没有介绍过的zval结构中的refcount字段了。
refcount,
顾名思义,记录了当前的zval被引用的计数。
比如对于代码:

1.       <?php

2.          $var = 1;

3.          $var_dup = $var;

4.       ?>

第一行,创建了一个整形变量,变量值是1此时保存整形1的这个zvalrefcount1
第二行,创建了一个新的整形变量,变量也指向刚才创建的zval,并将这个zvalrefcount1,此时这个zvalrefcount2
PHP
提供了一个函数可以帮助我们了解这个过程debug_zval_dump:

1.       <?php

2.        $var = 1;

3.        debug_zval_dump($var);

4.        $var_dup = $var;

5.        debug_zval_dump($var);

6.       ?>

输出:

1.       long(1) refcount(2)

2.       long(1) refcount(3)

如果你奇怪varrefcount应该是1啊?
我们知道,对于简单变量,PHP是以传值的形式穿参数的。也就是说,当执行debug_zval_dump($var)的时候,$var会以传值的方式传递给debug_zval_dump,也就是会导致varrefcount1,所以我们只要能看到,当变量赋值给一个变量以后,能导致zvalrefcount1这个事实即可。

现在我们回头看文章开头的代码,当执行了最后一行unset($var)以后,会发生什么呢?对,既是refcount1,上代码:

1.       <?php

2.          $var = "laruence";

3.          $var_dup = $var;

4.          unset($var);

5.          debug_zval_dump($var_dup);

6.       ?>

输出:

1.       string(8) "laruence" refcount(2)

但是,对于下面的代码呢?

1.       <?php

2.          $var = "laruence";

3.          $var_dup = $var;

4.          $var = 1;

5.       ?>

很明显在这段代码执行以后,$var_dup的值应该还是”laruence”, 那么这又是怎么实现的呢?
这就是PHPcopy on write机制:
PHP
在修改一个变量以前,会首先查看这个变量的refcount,如果refcount大于1PHP就会执行一个分离的例程,对于上面的代码,当执行到第三行的时候,PHP发现$var指向的zvalrefcount大于1,那么PHP就会复制一个新的zval出来,将原zvalrefcount1,并修改symbol_table,使得$var$var_dup分离(Separation)。这个机制就是所谓的copy on write(写时复制)

上代码测试:

1.       <?php

2.          $var = "laruence";

3.          $var_dup = $var;

4.          $var = 1;

5.          debug_zval_dump($var);

6.          debug_zval_dump($var_dup);

7.       ?>

输出:

1.       long(1) refcount(2)

2.       string(8) "laruence" refcount(2)

现在我们知道,当使用变量复制的时候PHP内部并不是真正的复制,而是采用指向相同的结构来尽量节约开销。那么,对于PHP中的引用,那又是如何实现呢?

1.       <?php

2.          $var = "laruence";

3.          $var_ref = &$var;

4.          $var_ref = 1;

5.       ?>

这段代码结束以后,$var也会被间接的修改为1,这个过程称作(change on write:写时改变)。那么ZE是怎么知道,这次的复制是不需要Separation的呢?
这个时候就要用到zval中的is_ref字段了:
对于上面的代码,当第二行执行以后,$var所代表的zvalrefcount变为2,并且同时置is_ref1
到第三行的时候,PHP先检查var_ref代表的zvalis_ref字段,如果为1,则不分离,大体逻辑示意如下:

1.        if((*val)->is_ref || (*val)->refcount<2){

2.                 //不执行Separation

3.               ... ;//process

4.         }

但是,问题又来了,对于如下的代码,又会怎样呢?

1.       <?php

2.          $var = "laruence";

3.          $var_dup = $var;

4.          $var_ref = &$var;

5.       ?>

对于上面的代码,存在一对copy on write的变量$var$var_dup, 又有一对change on write机制的变量对$var$var_ref,这个情况又是如何运作的呢?

当第二行执行的时候,和前面讲过的一样,$var_dup $var 指向相同的zval refcount2.
当执行第三行的时候,PHP发现要操作的zvalrefcount大于1,则,PHP会执行Separation, $var_dup分离出去,并将$var$var_refchange on write关联。也就是,refcount=2,is_ref=1;

基于这样的分析,我们就可以让debug_zval_dumprefcount1的结果来:

1.       <?php

2.            $var = "laruence";

3.           $var_dup = &$var;

4.            debug_zval_dump($var);

5.       ?>

输出:

1.       string(8) "laruence" refcount(1)

详细原因,读者你只要稍加分析就能得出,我就不越俎代庖了。;)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值