全面解读php-引用变量(&)
本文讲述引用传值的核心原理,看完即可扫清一切和引用传值相关的内容,不会了记得画图。
一、memory_get_usage的使用
传值赋值
// 定义一个变量
$a = range(0, 10000);
//memory_get_usage() 可以查看PHP内存使用量
var_dump(memory_get_usage()); // int(989778) // 定义变量b,将a变量的值赋值给b
$b = $a;
var_dump(memory_get_usage()); // int(989764) // 对a进行修改,只有发生了COW机制的才会重新开辟一块儿存储空间
// COW机制: Copy-On-Write
$a = range(0, 10000);
var_dump(memory_get_usage()); //int(1855643)
原理说明:
1、定义一个变量$a=range(0,1000);
;

2、$b = $a;
3、对a进行修改 $a = range(0, 10000)
COW机制
PHP写时复制机制(Copy-on-Write,也缩写为COW)
- 顾名思义,就是在写入时才真正复制一份内存进行修改。
- COW最早应用在Unix系统中对线程与内存使用的优化,后面广泛的被使用在各种编程语言中,如C++的STL等。
- 在PHP内核中,COW也是主要的内存优化手段。
- 在通过变量赋值的方式赋值给变量时,不会申请新内存来存放新变量的值,而是简单的通过一个计数器来共用内存。只有在其中的一个引用指向变量的值发生变化时,才申请新空间来保存值内容,以减少对内存的占用。
- 在很多场景下PHP都使用COW进行内存的优化。比如:变量的多次赋值、函数参数传递,并在函数体内修改实参等。
引用赋值
// 定义一个变量
$a = range(0, 10000);
var_dump(memory_get_usage()); //int(989761) // 定义变量b,将a变量的引用赋给b
$b = &$a;
var_dump(memory_get_usage()); //int(989876) // 对a进行修改
$a = range(0, 10000);
var_dump(memory_get_usage()); int(989869)
原理说明:
定义一个变量
$a = range(0, 10000);
定义变量b,将a变量的引用赋给b
$b = &$a;
对a进行修改
$a = range(0, 10000);
二、xdebug_debug_zval() 用于显示变量的信息
传值赋值
$a = 1;
//xdebug_debug_zval() 用于显示变量的信息。需要安装xdebug扩展。
xdebug_debug_zval('a'); //a:(refcount=1, is_ref=0) // 定义变量b,把a的值赋值给b
$b = $a;
xdebug_debug_zval('a');//a: (refcount=2, is_ref=0)
xdebug_debug_zval('b');//b: (refcount=2, is_ref=0) // a进行写操作
$a = 2;
xdebug_debug_zval('a');//a: (refcount=1, is_ref=0)
xdebug_debug_zval('b');//b: (refcount=1, is_ref=0)
1、定义变量 $a = 1;
$a = 1;
xdebug_debug_zval('a'); //a:(refcount=1, is_ref=0)
refcount=1
表示该变量指向的内存地址的引用个数变为1is_ref=0
表示该变量不是引用
2、定义变量 $b
,把 $a
的值赋给 $b
, $b = $a;
// 定义变量b,把a的值赋值给b
$b = $a;
xdebug_debug_zval('a');//a: (refcount=2, is_ref=0)
xdebug_debug_zval('b');//b: (refcount=2, is_ref=0)
refcount=2
表示该变量指向的内存地址的引用个数变为2is_ref=0
表示该变量不是引用
3、对变量 $a
进行写操作 $a = 2;
// a进行写操作
$a = 2;
xdebug_debug_zval('a');//a: (refcount=1, is_ref=0)
xdebug_debug_zval('b');//b: (refcount=1, is_ref=0)

因为COW机制,对变量 $a
进行写操作时,会为变量 $a
新分配一块内存空间,用于存储变量 $a
的值。
此时 $a
和 $b
指向的内存地址的引用个数都变为1。
引用赋值
$a = 1;
xdebug_debug_zval('a');//a: (refcount=1, is_ref=0) // 定义变量b,把a的引用赋给b
$b = &$a;
xdebug_debug_zval('a');//a: (refcount=2, is_ref=1)
xdebug_debug_zval('b');//b: (refcount=2, is_ref=1 // a进行写操作
$a = 2;
xdebug_debug_zval('a');//a: (refcount=2, is_ref=1)
xdebug_debug_zval('b');//b: (refcount=2, is_ref=1)
原理说明:
定义变量 $a = 1;
$a = 1;
xdebug_debug_zval('a');//a: (refcount=1, is_ref=0)
refcount=1
表示该变量指向的内存地址的引用个数变为1is_ref=0
表示该变量不是引用
2、定义变量 $b
,把 $a
的引用赋给 $b
, $b = &$a;
// 定义变量b,把a的引用赋给b
$b = &$a;
xdebug_debug_zval('a');//a: (refcount=2, is_ref=1)
xdebug_debug_zval('b');//b: (refcount=2, is_ref=1
refcount=2
表示该变量指向的内存地址的引用个数变为2is_ref=1
表示该变量是引用
3、对变量 $a
进行写操作 $a = 2;
// a进行写操作
$a = 2;
xdebug_debug_zval('a');//a: (refcount=2, is_ref=1)
xdebug_debug_zval('b');//b: (refcount=2, is_ref=1)
- 因为变量
$a
和变量$b
指向相同的内存地址,其实引用。 - 对变量
$a
进行写操作时,会直接修改指向的内存空间的值,因此变量$b
的值会跟着一起改变。
三、当变量时引用时,unset()只会取消引用,不会销毁内存空间
$a = 1;
$b = &$a; // unset 只会取消引用,不会销毁内存空间
unset($b); echo $a; //
原理说明:
定义变量 $a
,并将 $a
的引用赋给变量 $b
$a = 1;
$b = &$a;
销毁 $b
unset($b);
输出 $a
虽然销毁的 $b
,但是 $a
的引用和内存空间依旧存在。
echo $a; //
四、php中对象本身就是引用赋值
class Person
{
public $name = 'zhangsan';
} $p1 = new Person;
xdebug_debug_zval('p1'); //p1: (refcount=1, is_ref=0)=class Person { public $name = (refcount=2, is_ref=0)=1 } $p2 = $p1;
xdebug_debug_zval('p1');// p1: (refcount=2, is_ref=0)=class Person { public $name= (refcount=2, is_ref=0)=1 }
xdebug_debug_zval('p2'); //p2: (refcount=2, is_ref=0)=class Person { public $name= (refcount=2, is_ref=0)=1 } $p2->age = 2;
xdebug_debug_zval('p1');//p1: (refcount=2, is_ref=0)=class Person { public $name= (refcount=1, is_ref=0)=2 }
xdebug_debug_zval('p2');//p2: (refcount=2, is_ref=0)=class Person { public $name= (refcount=1, is_ref=0)=2 }
原理说明
1、实例化对象 $p1 = new Person;
$p1 = new Person;
xdebug_debug_zval('p1'); //p1: (refcount=1, is_ref=0)=class Person { public $name = (refcount=2, is_ref=0)=1 }
refcount=1
表示该变量指向的内存地址的引用个数变为1is_ref=0
表示该变量不是引用
2、把 $p1
赋给 $p2
$p2 = $p1;
xdebug_debug_zval('p1');// p1: (refcount=2, is_ref=0)=class Person { public $name= (refcount=2, is_ref=0)=1 }
xdebug_debug_zval('p2'); //p2: (refcount=2, is_ref=0)=class Person { public $name= (refcount=2, is_ref=0)=1 }
refcount=2
表示该变量指向的内存地址的引用个数变为2
3、对 $p2
中的属性 name 进行写操作
$p2->name = 'lisi';
xdebug_debug_zval('p1');//p1: (refcount=2, is_ref=0)=class Person { public $name= (refcount=1, is_ref=0)=2 }
xdebug_debug_zval('p2');//p2: (refcount=2, is_ref=0)=class Person { public $name= (refcount=1, is_ref=0)=2 }
因为php中对象本身就是引用赋值。对 $p2
中的属性 name 进行写操作时,会直接修改指向的内存空间的值,因此变量 $p1
的 name 属性的值会跟着一起改变。
五、实战例题分析
写出如下程序的输出结果 $d = ['a', 'b', 'c']; foreach($d as $k => $v)
{
$v = &$d[$k];
} 程序运行时,每一次循环结束后变量 $d 的值是什么?请解释。
程序执行完成后,变量 $d 的值是什么?请解释。
1. 第一次循环
推算出进入 foreach
时 $v
、$d[$k]
的值
$k = 0
$v = 'a'
$d[$k] = $d[0] = 'a'
此时,$v
和 $d[0]
在内存中分别开辟了一块空间
$v 和 $d[0] 在内存中分别开辟了一块空间
$v = &$d[0]
改变了 $v 指向的内存地址
$v = &$d[0]
$v = &$d[0] 改变了 $val 指向的内存地址
第一次循环后 $d 的值:
['a', 'b', 'c']
2. 第二次循环
进入 foreach
时 $v
被赋值为 'b',此时$v
指向的内存地址与 $d[0]
相同,且为引用,因此 $d[0]
的值被修改为 'b'
$v = 'b' => $d[0] = 'b' $v = ‘b’ => $d[0] = ‘b’
推算出进入 foreach
时 $d[$k]
的值
$k = 1
$d[$k] = $d[1] = 'b'
$d[2] = ‘b’
$v = &$d[1]
改变了 $v 指向的内存地址
$v = &$d[1]
$v = &$d[1]
第二次循环后 $d
的值
['b', 'b', 'c']
3. 第三次循环
进入 foreach
时 $v
被赋值为 'c',此时$v
指向的内存地址与 $d[1]
相同,且为引用,因此 $d[1]
的值被修改为 'c'
$v = 'c' => $d[1] = 'c' $v = ‘c’ => $d[1] = ‘c’]
推算出进入 foreach
时 $d[$k]
的值
$k = 2
$d[2] = 'c'
$d[2] = ‘c’
$v = &$d[2]
改变了 $v 指向的内存地址
$v = &$d[2]
$v = &$d[2]
第三次循环后 $d
的值
['b', 'c', 'c']
4. 实测
$d = ['a', 'b', 'c']; foreach ($d as $k=>$v)
{
$v = &$d[$k];
print_r($d);
} print_r($d);
Array
(
[0] => a
[1] => b
[2] => c
)
Array
(
[0] => b
[1] => b
[2] => c
)
Array
(
[0] => b
[1] => c
[2] => c
)
Array
(
[0] => b
[1] => c
[2] => c
)
`
全面解读php-引用变量(&)的更多相关文章
- Java基础-被final修饰的引用变量的指向
final修饰的引用变量一旦初始化赋值之后就不能再指向其他的对象,那么该引用变量指向的对象的内容可变吗?看下面这个例子: public class Test { public static void ...
- PHP GC垃圾回收机制之引用变量回收周期疑问
普通的引用变量的销毁大家都知道, 当unset的时候如果refcount = 0 则认为无用, 销毁. 但是手册中提到一点会有递归引用的问题,很是奇葩 代码如下 <?php $a = 1; $a ...
- C++引用变量(转)
引用变量 c++中引用变量的使用: 定义: int rate=80; int & pt=rate 1.pt 是引用变量,申明引用变量时必须将其初始化.pt 和rate 的值指向相同的内存变量 ...
- c#问答篇:对象与引用变量-----初学者的困惑
转自:http://www.cnblogs.com/huangyu/archive/2004/08/02/29622.html 从宏观的角度来看,对象是类的实例.比如: //定义一个名为Someone ...
- C++ primer(八)--内联函数 引用变量 引用传递函数参数 函数重载/模板/模板具体化
一.内联函数 常规函数和内联函数的区别在于C++编译器如何将他们组合到程序中.编译过程的最终产品是可执行程序--由一组机器语言指令组成.运行程序时,操作系统将这些指令载入到计算机内存中,因此每 ...
- C++引用变量学习
版权所有,转载请注明来源 (1)reference variable(rv) 主要用处是作为方程的形式参数,使用rv 可以直接对原数据进行操作而不是该数据的拷贝,节省了时间和空间,尤其是对于结构体以及 ...
- C++学习笔记29,引用变量(1)
引用变量在创建的时候就必须初始化.无法创建一个未被初始化的引用. #include <iostream> using namespace std; int main() { int x=1 ...
- [DB][mybatis]MyBatis mapper文件引用变量#{}与${}差异
MyBatis mapper文件引用变量#{}与${}差异 默认,使用#{}语法,MyBatis会产生PreparedStatement中.而且安全的设置PreparedStatement參数,这个过 ...
- PHP关于foreach使用引用变量的坑
写PHP好多年,但仍然会犯低级错误,今天遇到个 foreach中引用变量时的坑,PHP版本为 5.6.12 代码如下: <?php $arr = ['a', 'b', 'c', 'd', 'e' ...
- sed中引用变量
sed 中引用变量 eval sed 's/string/$REPLACE/g' file awk 中引用变量 awk 在匹配字符串的时候,有时候需要需要引用变量. $pid= eval " ...
随机推荐
- Slimvoice能代替JavaScript?
对于Slimvoice(https://slimvoice.co/),我想反对JavaScript的炒作,并对整个应用程序进行服务器端渲染.您可能会说:“用户必须在使用应用程序时重新加载每个页面,这必 ...
- 多线程的些许理解(平台x86,具体考虑linux,windows)
多线程的些许理解 一.体系架构 1.原子操作 1) 定义 不可中断的一个或者一系列操作,也就是不会被线程调度机制打断的操作,在运行期间不会有任何的上下文切换(context switch). 2) 我 ...
- linux下sendmail邮件系统安装详情
介绍 sendmail是linux系统中一个邮箱系统,如果我们在系统中配置好sendmail就可以直接使用它来发送邮箱.sendmail的配置文件/etc/mail/sendmail.cf ...
- linux命令详解——ftp
ftp服务器在网上较为常见,Linux ftp命令的功能是用命令的方式来控制在本地机和远程机之间传送文件,这里详细介绍Linux ftp命令的一些经常使用的命令,相信掌握了这些使用Linux 进行ft ...
- nfs服务的配置
nfs服务 nfs简介 Network file system 网络文件系统.NFS server可以看作是一个 file server.它可以让你的pc通过网络将远端的nfs server共享出来的 ...
- 青风nrf52832跑zephyr——点亮LED
zephyr版本:1.10 硬件:采用青风nrf52832开发板 开发环境:虚拟机Ubuntu16.04编译+Windows7 64bit烧录 使用的是 zephyr-zephyr-v1.10.0 ...
- Qualcomm_Mobile_OpenCL.pdf 翻译-1
1 前言 1.1 目的 这篇文档的主要目的是,向原始设备制造商(OEMs),独立软件供应商(ISVs),第三方开发者们,提供在基于高通骁龙400系列.600系列,和800系列的手机平台和芯片上进行开发 ...
- docker 创建容器时指定容器ip
Docker创建容器时默认采用bridge网络,自行分配ip,不允许自己指定. 在实际部署中,我们需要指定容器ip,不允许其自行分配ip,尤其是搭建集群时,固定ip是必须的. 我们可以创建自己的bri ...
- P1879 [USACO06NOV]玉米田Corn Fields[轮廓线DP]
状压暴力显然可做.但是数据出的再大一点就要稳T了.理论$O(n4^m)$,只不过实际跑不满. 考虑用轮廓线DP,设$f(i,j,S)$为处理到$(i,j)$时候(这格还不确定)的轮廓线为$S$的情况( ...
- P2197 【模板】nim游戏
博弈初心者... 学习地址luogu上可以找到.关于比较好的证明地址放在了地址页里了.这里不再赘述. 大概感觉还是所谓先手必胜就是面对当前局面一定可以采取一种策略,然后后手无论再怎么做,先手都可以“控 ...