2014-09-05
PHP语法中有两种赋值方式:引用赋值、非引用赋值。
<?php
$a = 1;
$b = $a; // 非引用赋值
$c = &$b; // 引用赋值
<?php
$a = 1;
$b = $a; // 非引用赋值
$c = &$b; // 引用赋值
从表面看,通常会这样认为:“引用赋值就是两个变量对应同一个变量(在C中其实就是一个zval),非引用赋值则是直接产生的一个新的变量(zval),同时将值copy过来”。
这种认为在大部分情况下都是可以想通的。(#1)
但有些情况下则会显得非常低效,例如:(#2)
<?php
function print_arr($arr){//非引用传递
print_r($arr);
}
$test_arr = array(
'a' => 'a',
'b' => 'b',
'c' => 'c',
...
);//这里一个比较大的数组
print_arr($test_arr);//第一次调用print_arr函数执行输出
print_arr($test_arr);//第二次调用print_arr函数执行输出
<?php
function print_arr($arr){//非引用传递
print_r($arr);
}
$test_arr = array(
'a' => 'a',
'b' => 'b',
'c' => 'c',
...
);//这里一个比较大的数组
print_arr($test_arr);//第一次调用print_arr函数执行输出
print_arr($test_arr);//第二次调用print_arr函数执行输出
如果按照上面的理解方式(#1),那么执行两次print_arr,并且是非引用的方式,则会产生两个与$test_arr完全相同的新的变量,那么将是非常低效的。
实际代码在运行中,并不会产生两个新的变量。因为PHP内核中已经帮助我们进行了优化。
具体如何实现的呢?这里就要讲到本文的要点:Reference counting & Copy-on-Write,正是采用引用计数、写时复制这两个机制得以优化。
在介绍这两个机制前,先了解一个基本知识:PHP中的变量在内核中是如何表示的。
PHP中定义的变量都是以一个zval来表示的,zval的定义在Zend/zend.h中定义:
typedef struct _zval_struct zval;
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;
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;
};
其中,refcount和is_ref就是实现引用计数、写时复制这两个机制的基础。
refcount当前变量存储引用计数,在zval初始创建的时候就为1。每增加一个引用,则refcount ++。当进行引用分离时,refcount--。
is_ref用于表示一个zval是否是引用状态。zval初始化的情况下会是0,表示不是引用。
<?php
$a;//a:refcount=1,is_ref=0, value=NULL;
$a = 1; //a:refcount=2,is_ref=0, value=1;
$b = $a; //a,b:refcount=3,is_ref=0,value=1;
$c = $a; //a,b,c:refcount=4,is_ref=0,value=1;
$d = &$c; //a,b:refcount=3,is_ref=0,value=1; c,d:refcount=1, is_ref=1, value=1
<?php
$a;//a:refcount=1,is_ref=0, value=NULL;
$a = 1; //a:refcount=2,is_ref=0, value=1;
$b = $a; //a,b:refcount=3,is_ref=0,value=1;
$c = $a; //a,b,c:refcount=4,is_ref=0,value=1;
$d = &$c; //a,b:refcount=3,is_ref=0,value=1; c,d:refcount=1, is_ref=1, value=1上面代码的注释,表示当执行这一行后,refcount与is_ref的变化.
Copy on Write
Php变量通过引用计数实现变量共享数据,那如果改变其中一个变量值呢?
当试图写入一个变量时,Zend若发现该变量指向的zval被多个变量共享,则为其复制一份ref_count为1的zval,并递减原zval的refcount,这个过程称为“zval分离”。可见,只有在有写操作发生时zend才进行拷贝操作,因此也叫copy-on-write(写时拷贝)
对于引用型变量,其要求和非引用型相反,引用赋值的变量间必须是捆绑的,修改一个变量就修改了所有捆绑变量。
<?php
$a=1;
$b=$a;
<?php
$a=1;
$b=$a;执行过程中的内存结构图:
<?php
$a=1;
$b=&a;
<?php
$a=1;
$b=&a;执行过程中的内存结构图:
从上可以看到,无论是引用、非引用,这种直接赋值都不会产生新的变量。
只是当是引用时,is_ref设置为1。当非引用时,is_ref设置为0。
读写复制,就是根据is_ref来进行变量分离的。
当is_ref=1时,是引用变量时,执行“引用下的变量分离”
<?php
$a = 1;
$b = $a;
$c = &$b;
<?php
$a = 1;
$b = $a;
$c = &$b;执行过程中的内存结构图:
当is_ref=0时,是非引用变量时,执行“非引用下的变量分离”
<?php
$a = 1;
$b = &$a;
$c = $b;
<?php
$a = 1;
$b = &$a;
$c = $b;
执行过程中的内存结构图:
只有真正在需要改变变量的值时,
回头在看(#2)代码,可以看到实际上,并没有产生新的变量,始终是$test_arr的变量在输出。所以,这也是为什么很少看到在PHP中使用引用方式传递变量,却仍然不会有性能问题的原因。
1
CI框架连接数据库配置操作以及多数据库操作
09-05
2
asp 简单读取数据表并列出来 ASP如何快速从数据库读取大量数据
05-17
3
C语言关键字及其解释介绍 C语言32个关键字详解
04-05
4
C语言中sizeof是什么意思 c语言里sizeof怎样用法详解
04-26
5
最简单的asp登陆界面代码 asp登陆界面源代码详细介绍
04-12
6
PHP中的魔术方法 :__construct, __destruct , __call, __callStatic,__get, __set, __isset, __unset , __sleep,
09-05
7
PHP中的(++i)前缀自增 和 (i++)后缀自增
09-05
8
PHP中include和require区别之我见
09-05
常用dos命令及语法
2014-09-27
将视频设置为Android手机开机动画的教程
2014-12-11
php递归返回值的问题
2014-09-05
如何安装PHPstorm并配置方法教程 phpstorm安装后要进行哪些配置
2017-05-03
java中的info是什么意思
2022-03-24
PHP 教程之如何使用BLOB存取图片信息实例
2014-09-05
IcePHP框架中的快速后台中的通用CRUD功能框架
2014-09-05
单片机编程好学吗?单片机初学者怎样看懂代码
2022-03-21
PHP数组函数array
2014-09-05
学ug编程如何快速入门?
2022-03-17
趣游捕鱼高爆版官方最新版下载v12.0 安卓版
休闲益智 120.2M
下载趣游捕鱼官方正版手游下载v12.0 安卓最新版
休闲益智 117.1M
下载永夜降临复苏游戏下载v3.5.7 安卓最新版
卡牌对战 537.6M
下载创世战车手游下载v1.45.1.98037 安卓版
射击枪战 2.36G
下载dokkan七龙珠爆裂激战国际版手游下载v5.28.6 安卓版
动作闯关 97.6M
下载这就是江湖官方版下载v12.6.6 安卓最新版本
卡牌对战 211.6M
下载热血美职篮手游下载v1.19020.1722.0 安卓版
体育运动 1.62G
下载斗罗大陆魂师对决手游官方版下载v2.31.1 安卓版
卡牌对战 1.55G
下载坦克大决战游戏下载v1.9462 安卓官方正版
下载
孤胆车神维加斯官方正版(Gangstar Vegas)下载v8.5.1c 安卓手机版
下载
孤胆车神维加斯僵尸城版本下载v8.5.1c 安卓版
下载
英雄联盟云顶之弈手机版本下载v14.8.5768838 安卓最新版本
下载
苍蓝前线手游下载v1.1.0 安卓版
下载
崩坏学园2手游下载v12.2.8 安卓最新版本
下载
彩虹联萌手游下载v7.3.23015 安卓官方版
下载
花花填色苹果版下载v2.2.5 iPhone版
下载