---
title: 理解PHP的变量,值与引用的关系
createdDate: 2015-03-11
category: php
---

PHP的变量与C++中的变量是两种截然不容的概念。如果没有理解清楚,使用C++的方式来思考PHP就会遇到一些问题。

C++中,变量与值是绑定的。值是内存的上的一块内存上的数据,而变量则是操作这块内存的名称。变量消失(比如超出作用域)值也会消失。

而PHP中,变量和值是两个概念。PHP是一种弱类型语言,值在PHP的内部(zend引擎),被存放在一个zval结构体中,这个结构体中,除了包含了值的类型,数据外,还包含两个字节的信息。一个是`is_ref`,是一个bool值,用来标识这个值是否是一个`引用`。第二个额外字节是`refcount`,用来表示指向这个值的变量(也称符号即symbol)的个数。如果refcount为0,那么这个值就可以被回收了。

而变量则另外存放在一个符号表中,每个变量有其作用域。

声明一个变量时:

```
$a = 'hello';
```

在当前作用域新建了一个变量a,并在内存中新建了一个zval,其类型字段为string,数据字段为'hello'。因为a不是一个引用变量,所以is_ref字段为false,因为$a指向了这个zval(注意,虽然是指向这个zval了,但是a依然不是引用变量),所以refcount为1。关系大概类似这样:



注意一点,**当"refcount"的值是1时,"is_ref"的值总是FALSE**。这一点往后就明白了。

## 输出"refcount"和"is_ref"

如果按照了xdebug,可以使用`xdebug_debug_zval()`函数来输出"refcount"和"is_ref"的值。

```
$a = 'hello';
xdebug_debug_zval('a');
```

输出:
```
(refcount=1, is_ref=0),string 'hello' (length=5)
```

我们可以利用这个函数来更好的理解zval。

## 赋值的小细节

先来看那PHP中赋值的例子:

```
$a = "hello";
$b = $a;
xdebug_debug_zval( 'a' );
xdebug_debug_zval( 'b' );
```

输出

```
a:
(refcount=2, is_ref=0),string 'hello' (length=5)
b:
(refcount=2, is_ref=0),string 'hello' (length=5)
```

既然PHP中除了object的赋值外都是复制,那么为什么a的refcount会增加呢?

我们修改b的值看看会不会有什么变化:

```
$a = "hello";
$b = $a;
xdebug_debug_zval( 'a' );
xdebug_debug_zval( 'b' );

$b = 2;
xdebug_debug_zval('a');
xdebug_debug_zval('b');
```

输出:
```
a:
(refcount=2, is_ref=0),string 'hello' (length=5)
b:
(refcount=2, is_ref=0),string 'hello' (length=5)
a:
(refcount=1, is_ref=0),string 'hello' (length=5)
b:
(refcount=1, is_ref=0),int 2
```

看到了么!如果只是赋值而没有修改的话,PHP不会立马拷贝值,而是让b指向a的值,所以a的zval的refcount为2。但是当我们修改b时,PHP就必须进行复制了。所以a的zval的refcount回到1。

用图片表示更直观:

$b = $a:



修改b后:



## 理解引用

理解了PHP中值与引用的关系,再来看PHP的引用就非常简单了。引用变量不会有新开一个zval,而是指向他所引用的变量的zval。并且,zval的`is_ref`置为1,refcount加一。

把上面的例子改为引用赋值,看看区别:
```
a:
(refcount=2, is_ref=1),string 'hello' (length=5)
b:
(refcount=2, is_ref=1),string 'hello' (length=5)
a:
(refcount=2, is_ref=1),int 2
b:
(refcount=2, is_ref=1),int 2
```

输出:

```
a:
(refcount=2, is_ref=1),string 'hello' (length=5)
b:
(refcount=2, is_ref=1),string 'hello' (length=5)
a:
(refcount=2, is_ref=1),int 2
b:
(refcount=2, is_ref=1),int 2
```

可以看出a和b都是指向一个zval的变量。



在代码的最后把a变量删除会有什么效果呢?

```
$a = "hello";
$b = &$a;
xdebug_debug_zval( 'a' );
xdebug_debug_zval( 'b' );

$b = 2;
xdebug_debug_zval('a');
xdebug_debug_zval('b');

unset($a);
xdebug_debug_zval('b');
```

输出:
```
a:
(refcount=2, is_ref=1),string 'hello' (length=5)
b:
(refcount=2, is_ref=1),string 'hello' (length=5)
a:
(refcount=2, is_ref=1),int 2
b:
(refcount=2, is_ref=1),int 2
b:
(refcount=1, is_ref=0),int 2
```

删除a变量后,b的zval的refcount减为1,这个可以理解,这里需要注意的是,b作为一个引用变量,现在他的zval的`is_ref`为0!也就是它不再是引用变量了。

这是因为a变量被删除后,b变为唯一一个指向这个zval的变量了,也因此,他也就摆脱了引用变量这个身份。这也就是上文所说的**当"refcount"的值是1时,"is_ref"的值总是FALSE**



## 参考网址
- PHP: 引用计数基本知识 - Manual
- PHP: 引用的解释 - Manual

理解PHP的变量,值与引用的关系的更多相关文章

  1. 理解--->Java中的值传递&引用传递

    转自:http://url.cn/5tL9F5D 值传递和引用传递 值传递(pass by value)是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际 ...

  2. Javascript参数传递中值和引用的一种理解

    值(value)和引用(reference)是各种编程语言老生常谈的话题,js也不例外. 我将剖析一个例子的实际运行过程,跟大家分享我对js参数传递中的值和引用的理解. 参考官网数据类型的两种分类,本 ...

  3. 【原创】深入理解c++的右值引用

    0 左值和右值     一个左值表达式代表的是对象本身,而右值表达式代表的是对象的值:变量也是左值.   1 右值引用作用 为了支持移动操作(包括移动构造函数和移动赋值函数),C++才引入了一种新的引 ...

  4. javascript值和引用

    JavaScript引用指向的是值. 简单值(即标量基本类型值,基本类型值,js中6类,null.undefined.boolean.number.string和symbol)总是通过值复制的方式来赋 ...

  5. PHP javascript 值互相引用(不用刷新页面)

    PHP javascript 值互相引用的问题   昨天通过EMAIL给一些公司投了简历,希望他们能给我一份工作,今天其中一家公司的人给我打电话,大意是要我做一点东西(与AJAX有关) 给他们看,我听 ...

  6. C++ : 从栈和堆来理解C#中的值类型和引用类型

    C++中并没有值类型和引用类型之说,标准变量或者自定义对象的存取默认是没有区别的.但如果深入地来看,就要了解C++中,管理数据的两大内存区域:栈和堆. 栈(stack)是类似于一个先进后出的抽屉.它的 ...

  7. JS中原始值和引用值的储存方式

    在ECMAscript中,变量可以存放两种类型的值,即原始值和引用值 原始值指的是代表原始数据类型的值,也叫基本数据类型,包括:Number.Stirng.Boolean.Null.Underfine ...

  8. C/C++-左值、右值及引用

    目录 1.左值and右值 2.引用 3.左值引用的用途 4.std::move和std::swap C和C++中定义了引用类型(reference type),存在左值引用(lvalue refere ...

  9. 理解Javascript的变量提升

    前言 本文2922字,阅读大约需要8分钟. 总括: 什么是变量提升,使用var,let,const,function,class声明的变量函数类在变量提升的时候都有什么区别. 参考文章:Hoistin ...

随机推荐

  1. Spring + MyBatis 多数据源实现

    近期,在项目中需要做分库,但是因为某些原因,没有采用开源的分库插件,而是采用了同事之前弄得多数据源形式实现的分库.对于多数据源,本人在实际项目也中遇到的不多,之前的项目大多是服务化,以RPC的形式获得 ...

  2. 辨析各类web服务器:Apache/Tomcat/Jboss/Nginx/等,还有Nodejs

    先说一下各类服务器能干啥,特点是啥,然后在区分他们的类别. (1)Apache: Apache是指Apache软件基金会的Apache HTTP Server, 它能够接收http请求,然后返回各类资 ...

  3. scp命令详解—跨服务器复制文件

    scp在跨机器复制的时候为了提高数据的安全性,使用了ssh连接和加密方式,如果机器之间配置了ssh免密码登录,那在使用scp的时候密码都不用输入. 在服务器104.238.161.75上操作,将服务器 ...

  4. python版本管理(python环境隔离)

    这将是一篇比较短的文章. 我发文向来注重文章质量,营养不够的宁可不发,但是我相信很多人需要这篇文章. 之所以要去搞清楚这个问题,是我在把 vscode 的 inspector 设置为 pipenv 生 ...

  5. CentOS7.5安装坚果云

    1.下载坚果云rpm包,对于CentOS,选fedora那个版本的包 https://www.jianguoyun.com/s/downloads/linux 2.安装 坚果云安装依赖notify-p ...

  6. Qt不同版本编译器,调用VC++生成的动态链接库

    今天用QT编译生成的共享库自己却怎么都不能调用,查了N久后找到这个帖子,发现搞定了,记录一下 http://qiusuoge.com/12720.html Qt如何调用VC++生成的动态链接库?假设当 ...

  7. Linux基础系列-Day2

    基础命令(文件内容管理) 1.cat:在当前终端显示文本文件内容 格式:cat [文件路径] -n 从1开始对所有输出的行数编号 -b 和-n相似,只不过对于空白行不编号:2.head:从文件内容开头 ...

  8. 高效的 itertools 模块(转)

    原文地址:http://python.jobbole.com/87380/ 我们知道,迭代器的特点是:惰性求值(Lazy evaluation),即只有当迭代至某个值时,它才会被计算,这个特点使得迭代 ...

  9. SQLSEVER 中的那些键和约束

    SQL Server中有五种约束类型,分别是 PRIMARY KEY约束.FOREIGN KEY约束.UNIQUE约束.DEFAULT约束.和CHECK约束.查看或者创建约束都要使用到 Microso ...

  10. ZOJ 2589 Circles(平面图欧拉公式)

    [题目链接] http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=2589 [题目大意] 给出一些圆,问这些圆可以把平面分为几个部 ...