浅析PHP原理之变量分离/引用(Variables Separation)


Posted in PHP onAugust 09, 2013

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

struct _zval_struct {
        /* Variable information */
        zvalue_value value; /* value */
        zend_uint refcount;
        zend_uchar type; /* active type */
        zend_uchar is_ref;
};

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

第一行代码创建了一个字符串变量,申请了一个大小为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被引用的计数。
比如对于代码:
<?php
   $var = 1;
   $var_dup = $var;
?>

第一行,创建了一个整形变量,变量值是1。 此时保存整形1的这个zval的refcount为1。
第二行,创建了一个新的整形变量,变量也指向刚才创建的zval,并将这个zval的refcount加1,此时这个zval的refcount为2。
PHP提供了一个函数可以帮助我们了解这个过程debug_zval_dump:
<?php
 $var = 1;
 debug_zval_dump($var);
 $var_dup = $var;
 debug_zval_dump($var);
?>
输出:
long(1) refcount(2)
long(1) refcount(3

如果你奇怪 ,var的refcount应该是1啊?
我们知道,对于简单变量,PHP是以传值的形式穿参数的。也就是说,当执行debug_zval_dump($var)的时候,$var会以传值的方式传递给debug_zval_dump,也就是会导致var的refcount加1,所以我们只要能看到,当变量赋值给一个变量以后,能导致zval的refcount加1这个事实即可。
现在我们回头看文章开头的代码, 当执行了最后一行unset($var)以后,会发生什么呢? 对,既是refcount减1,上代码:
<?php
   $var = "laruence";
   $var_dup = $var;
   unset($var);
   debug_zval_dump($var_dup);
?>
输出:
string(8) "laruence" refcount(2

但是,对于下面的代码呢?
<?php
   $var = "laruence";
   $var_dup = $var;
   $var = 1;
?>

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

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

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

但是,问题又来了,对于如下的代码,又会怎样呢?
<?php
   $var = "laruence";
   $var_dup = $var;
   $var_ref = &$var;
?>

对于上面的代码,存在一对copy on write的变量$var和$var_dup, 又有一对change on write机制的变量对$var和$var_ref,这个情况又是如何运作的呢?
当第二行执行的时候,和前面讲过的一样,$var_dup 和 $var 指向相同的zval, refcount为2.
当执行第三行的时候,PHP发现要操作的zval的refcount大于1,则,PHP会执行Separation, 将$var_dup分离出去,并将$var和$var_ref做change on write关联。也就是,refcount=2, is_ref=1;
基于这样的分析,我们就可以让debug_zval_dump出refcount为1的结果来:
<?php
     $var = "laruence";
    $var_dup = &$var;
     debug_zval_dump($var);
?>
输出:
string(8) "laruence" refcount(1

详细原因,读者你只要稍加分析就能得出,我就不越俎代庖了。;)
这次我们介绍了PHP的变量分离机制,下次我会继续介绍如果在扩展中接收和传出PHP脚本中的参数。
PHP 相关文章推荐
PHP的FTP学习(一)[转自奥索]
Oct 09 PHP
小偷PHP+Html+缓存
Nov 25 PHP
php 安全过滤函数代码
May 07 PHP
PHP获取文件绝对路径的代码(上一级目录)
May 29 PHP
php发送post请求的三种方法
Feb 11 PHP
PHP数据库万能引擎类adodb配置使用以及实例集锦
Jun 12 PHP
推荐25款php中非常有用的类库
Sep 29 PHP
自己写的php中文截取函数mb_strlen和mb_substr
Feb 09 PHP
php通过function_exists检测函数是否存在的方法
Mar 18 PHP
PHP函数func_num_args用法实例分析
Dec 07 PHP
CI(CodeIgniter)模型用法实例分析
Jan 20 PHP
php菜单/评论数据递归分级算法的实现方法
Aug 01 PHP
php中如何使对象可以像数组一样进行foreach循环
Aug 09 #PHP
php接口与接口引用的深入解析
Aug 09 #PHP
解析数组非数字键名引号的必要性
Aug 09 #PHP
php防注入及开发安全详细解析
Aug 09 #PHP
分割GBK中文遭遇乱码的解决方法
Aug 09 #PHP
解析isset与is_null的区别
Aug 09 #PHP
PHP中怎样保持SESSION不过期 原理及方案介绍
Aug 08 #PHP
You might like
PHP header函数分析详解
2011/08/06 PHP
js判断浏览器的比较全的代码
2007/02/13 Javascript
js 全兼容可高亮二级缓冲折叠菜单
2010/06/04 Javascript
js相册效果代码(点击创建即可)
2013/04/16 Javascript
js中prototype用法详细介绍
2013/11/14 Javascript
按钮接受回车事件的三种实现方法
2014/06/06 Javascript
jQuery选择器源码解读(四):tokenize方法的Expr.preFilter
2015/03/31 Javascript
Javascript基础知识盲点总结之函数
2016/05/15 Javascript
AngularJS基础 ng-if 指令用法
2016/08/01 Javascript
浅谈js对象属性 通过点(.) 和方括号([]) 的不同之处
2016/10/29 Javascript
Express框架之connect-flash详解
2017/05/31 Javascript
基于Bootstrap分页的实例讲解(必看篇)
2017/07/04 Javascript
vue.js移动端app实战1:初始配置详解
2017/07/24 Javascript
JQuery元素快速查找与操作
2018/04/22 jQuery
JS实现常见的查找、排序、去重算法示例
2018/05/21 Javascript
JS/HTML5游戏常用算法之碰撞检测 包围盒检测算法详解【圆形情况】
2018/12/13 Javascript
vue router动态路由设置参数可选问题
2019/08/21 Javascript
JS实现动态星空背景效果
2019/11/01 Javascript
[50:34]VGJ.T vs Fnatic 2018国际邀请赛小组赛BO2 第二场 8.16
2018/08/17 DOTA
详解Python中的array数组模块相关使用
2016/07/05 Python
python pygame实现五子棋小游戏
2020/10/26 Python
基于python实现的百度音乐下载器python pyqt改进版(附代码)
2019/08/05 Python
PyTorch之图像和Tensor填充的实例
2019/08/18 Python
在vscode中配置python环境过程解析
2019/09/28 Python
python实现的按要求生成手机号功能示例
2019/10/08 Python
pip安装提示Twisted错误问题(Python3.6.4安装Twisted错误)
2020/05/09 Python
Python使用matplotlib绘制圆形代码实例
2020/05/27 Python
解决Pycharm 中遇到Unresolved reference 'sklearn'的问题
2020/07/13 Python
Selenium之模拟登录铁路12306的示例代码
2020/07/31 Python
戴尔美国官网:Dell
2016/08/31 全球购物
华润集团网上药店:健一网
2016/09/19 全球购物
美国沃尔玛网上超市:Walmart
2020/08/14 全球购物
金融系毕业生自荐书
2014/07/08 职场文书
运动会广播稿200米(5篇)
2014/10/15 职场文书
解除同居协议书
2015/01/29 职场文书
Python 发送SMTP邮件的简单教程
2021/06/24 Python