深入PHP变量存储的详解


Posted in PHP onJune 13, 2013

1.1.1 zval结构
Zend使用zval结构来存储PHP变量的值,该结构如下所示:

typedef union _zvalue_value {
 long lval;    /* long value */
 double dval;    /* double value */
 struct {
  char *val;
  int len;
 } str;
 HashTable *ht;    /* hash table value */
 zend_object_value obj;
} zvalue_value;

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

Zend根据type值来决定访问value的哪个成员,可用值如下:
IS_NULL N/A
IS_LONG 对应value.lval
IS_DOUBLE 对应value.dval
IS_STRING 对应value.str
IS_ARRAY 对应value.ht
IS_OBJECT 对应value.obj
IS_BOOL 对应value.lval.
IS_RESOURCE 对应value.lval

根据这个表格可以发现两个有意思的地方:首先是PHP的数组其实就是一个HashTable,这就解释了为什么PHP能够支持关联数组了;其次,Resource就是一个long值,它里面存放的通常是个指针、一个内部数组的index或者其它什么只有创建者自己才知道的东西,可以将其视作一个handle。

1.1.2 引用计数
引用计数在垃圾收集、内存池以及字符串等地方应用广泛,Zend就实现了典型的引用计数。多个PHP变量可以通过引用计数机制来共享同一份zval,zval中剩余的两个成员is_ref和refcount就用来支持这种共享。
很明显,refcount用于计数,当增减引用时,这个值也相应的递增和递减,一旦减到零,Zend就会回收该zval。
那么is_ref呢?

1.1.3 zval状态
在PHP中,变量有两种——引用和非引用的,它们在Zend中都是采用引用计数的方式存储的。对于非引用型变量,要求变量间互不相干,修改一个变量时,不能影响到其他变量,采用Copy-On-Write机制即可解决这种冲突——当试图写入一个变量时,Zend若发现该变量指向的zval被多个变量共享,则为其复制一份refcount为1的zval,并递减原zval的refcount,这个过程称为“zval分离”。然而,对于引用型变量,其要求和非引用型相反,引用赋值的变量间必须是捆绑的,修改一个变量就修改了所有捆绑变量。
可见,有必要指出当前zval的状态,以分别应对这两种情况,is_ref就是这个目的,它指出了当前所有指向该zval的变量是否是采用引用赋值的——要么全是引用,要么全不是。此时再修改一个变量,只有当发现其zval的is_ref为0,即非引用时,Zend才会执行Copy-On-Write。

1.1.4 zval状态切换
当在一个zval上进行的所有赋值操作都是引用或者都是非引用时,一个is_ref就足够应付了。然而,世界总不会那么美好,PHP无法对用户进行这种限制,当我们混合使用引用和非引用赋值时,就必须要进行特别处理了。
情况I、看如下PHP代码:

<?php
$a = 1;
$b = $a;
$c = $b;
$d = &$c; // 在一堆非引用赋值中,插入一个引用
?>

这段代码首先进行了一次初始化,这将创建一个新的zval,is_ref=0, refcount=1,并将a指向这个zval;之后是两次非引用赋值,正如前面所说,只要把b和c都指向a的zval即可;最后一行是个引用赋值,需要is_ref为1,但是Zend发现c指向的zval并不是引用型的,于是为c创建单独的zval,并同时将d指向该zval。
从本质上来说,这也可以看作是一种Copy-On-Write,不仅仅是value,is_ref也是受保护的对象。
整个过程图示如下:

深入PHP变量存储的详解

情况II,看如下PHP代码:

<?php
$a = 1;
$b = &$a;
$c = &$b;
$d = $c; // 在一堆引用赋值中,插入一个非引用
?>

这段代码的前三句将把a、b和c指向一个zval,其is_ref=1, refcount=3;第四句是个非引用赋值,通常情况下只需要增加引用计数即可,然而目标zval属于引用变量,单纯的增加引用计数显然是错误的, Zend的解决办法是为d单独生成一份zval副本。
全过程如下所示:

深入PHP变量存储的详解

1.1.5 参数传递
PHP函数参数的传递和变量赋值是一样的,非引用传递相当于非引用赋值,引用传递相当于引用赋值,并且也有可能会导致执行zval状态切换。这在后面还将提到。

PHP 相关文章推荐
MySQL相关说明
Jan 15 PHP
PHP实现采集程序原理和简单示例代码
Mar 18 PHP
生成随机字符串和验证码的类的PHP实例
Dec 24 PHP
浅谈php正则表达式中的非贪婪模式匹配的使用
Nov 25 PHP
php中file_exists函数使用详解
May 08 PHP
PHP简单实现HTTP和HTTPS跨域共享session解决办法
May 27 PHP
PHP限制HTML内容中图片必须是本站的方法
Jun 16 PHP
使用PHP编写发红包程序
Jul 22 PHP
PHP中Trait及其应用详解
Feb 14 PHP
PHP基于rabbitmq操作类的生产者和消费者功能示例
Jun 16 PHP
yii2.0框架使用 beforeAction 防非法登陆的方法分析
Sep 11 PHP
PHP 对象接口简单实现方法示例
Apr 13 PHP
深入PHP中的HashTable结构详解
Jun 13 #PHP
基于PHP输出缓存(output_buffering)的深入理解
Jun 13 #PHP
php缓冲 output_buffering的使用详解
Jun 13 #PHP
如何在PHP中使用正则表达式进行查找替换
Jun 13 #PHP
php启用zlib压缩文件的配置方法
Jun 12 #PHP
Window下PHP三种运行方式图文详解
Jun 11 #PHP
控制PHP的输出:缓存并压缩动态页面
Jun 11 #PHP
You might like
php数组函数序列之prev() - 移动数组内部指针到上一个元素的位置,并返回该元素值
2011/10/31 PHP
PHP Laravel 上传图片、文件等类封装
2017/08/16 PHP
extJs 文本框后面加上说明文字+下拉列表选中值后触发事件
2009/11/27 Javascript
基于iframe实现类似于ajax的页面无刷新
2014/05/31 Javascript
javascript实现10个球随机运动、碰撞实例详解
2015/07/08 Javascript
js中对函数设置默认参数值的3种方法
2015/10/23 Javascript
jQuery实现二级下拉菜单效果
2016/01/05 Javascript
Bootstrap模态框水平垂直居中与增加拖拽功能
2016/11/09 Javascript
深入理解JavaScript中的尾调用(Tail Call)
2017/02/07 Javascript
JS实现登录页密码的显示和隐藏功能
2017/12/06 Javascript
JavaScript 预解析的4种实现方法解析
2019/09/03 Javascript
Element 默认勾选表格 toggleRowSelection的实现
2019/09/04 Javascript
微信小程序pinker组件使用实现自动相减日期
2020/05/07 Javascript
[02:36]DOTA2亚洲邀请赛小组赛精彩集锦:EE凭借法力虚空拿下4杀
2017/03/30 DOTA
[47:38]Optic vs VGJ.S 2018国际邀请赛小组赛BO2 第二场 8.17
2018/08/20 DOTA
python 筛选数据集中列中value长度大于20的数据集方法
2018/06/14 Python
python获取微信小程序手机号并绑定遇到的坑
2018/11/19 Python
Django对数据库进行添加与更新的例子
2019/07/12 Python
python 实现手机自动拨打电话的方法(通话压力测试)
2019/08/08 Python
python实现七段数码管和倒计时效果
2019/11/23 Python
Python实现快速大文件比较代码解析
2020/09/04 Python
python 如何利用argparse解析命令行参数
2020/09/11 Python
python爬虫---requests库的用法详解
2020/09/28 Python
css3.0 图形构成实例练习一
2013/03/19 HTML / CSS
乐天旅游台湾网站:Rakuten Travel TW
2017/06/01 全球购物
美国购买韩国护肤和美容产品网站:Althea Korea
2020/11/16 全球购物
个人找工作自荐信格式
2013/09/21 职场文书
室内设计实习自我鉴定
2013/09/25 职场文书
国际贸易专业个人求职信格式
2014/02/02 职场文书
学校后勤岗位职责
2014/02/19 职场文书
库房保管员岗位职责
2014/04/07 职场文书
幼儿评语大全
2014/04/30 职场文书
护理专业毕业生自荐信
2014/06/15 职场文书
内勤岗位职责
2015/02/10 职场文书
幼儿园见习总结
2015/06/23 职场文书
读《皮囊》有感:理解是对他人的最大的善举
2019/11/14 职场文书