本文讲述引用传值的核心原理,看完即可扫清一切和引用传值相关的内容,不会了记得画图。

一、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)

原理说明:

  1. 定义一个变量 $a = range(0, 10000);

  2. 定义变量b,将a变量的引用赋给b $b = &$a;

  3. 对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 表示该变量指向的内存地址的引用个数变为1
is_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 表示该变量指向的内存地址的引用个数变为2
is_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 表示该变量指向的内存地址的引用个数变为1
is_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 表示该变量指向的内存地址的引用个数变为2
is_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 表示该变量指向的内存地址的引用个数变为1
is_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-引用变量(&)的更多相关文章

  1. Java基础-被final修饰的引用变量的指向

    final修饰的引用变量一旦初始化赋值之后就不能再指向其他的对象,那么该引用变量指向的对象的内容可变吗?看下面这个例子: public class Test { public static void ...

  2. PHP GC垃圾回收机制之引用变量回收周期疑问

    普通的引用变量的销毁大家都知道, 当unset的时候如果refcount = 0 则认为无用, 销毁. 但是手册中提到一点会有递归引用的问题,很是奇葩 代码如下 <?php $a = 1; $a ...

  3. C++引用变量(转)

    引用变量 c++中引用变量的使用: 定义: int rate=80; int  & pt=rate 1.pt 是引用变量,申明引用变量时必须将其初始化.pt 和rate 的值指向相同的内存变量 ...

  4. c#问答篇:对象与引用变量-----初学者的困惑

    转自:http://www.cnblogs.com/huangyu/archive/2004/08/02/29622.html 从宏观的角度来看,对象是类的实例.比如: //定义一个名为Someone ...

  5. C++ primer(八)--内联函数 引用变量 引用传递函数参数 函数重载/模板/模板具体化

    一.内联函数     常规函数和内联函数的区别在于C++编译器如何将他们组合到程序中.编译过程的最终产品是可执行程序--由一组机器语言指令组成.运行程序时,操作系统将这些指令载入到计算机内存中,因此每 ...

  6. C++引用变量学习

    版权所有,转载请注明来源 (1)reference variable(rv) 主要用处是作为方程的形式参数,使用rv 可以直接对原数据进行操作而不是该数据的拷贝,节省了时间和空间,尤其是对于结构体以及 ...

  7. C++学习笔记29,引用变量(1)

    引用变量在创建的时候就必须初始化.无法创建一个未被初始化的引用. #include <iostream> using namespace std; int main() { int x=1 ...

  8. [DB][mybatis]MyBatis mapper文件引用变量#{}与${}差异

    MyBatis mapper文件引用变量#{}与${}差异 默认,使用#{}语法,MyBatis会产生PreparedStatement中.而且安全的设置PreparedStatement參数,这个过 ...

  9. PHP关于foreach使用引用变量的坑

    写PHP好多年,但仍然会犯低级错误,今天遇到个 foreach中引用变量时的坑,PHP版本为 5.6.12 代码如下: <?php $arr = ['a', 'b', 'c', 'd', 'e' ...

  10. sed中引用变量

    sed 中引用变量 eval sed 's/string/$REPLACE/g' file awk 中引用变量 awk 在匹配字符串的时候,有时候需要需要引用变量. $pid= eval " ...

随机推荐

  1. idea一键生成mybatis工具

    1.创建maven项目,导包 <build> <plugins> <plugin> <groupId>org.mybatis.generator< ...

  2. Mac OSX编译安装php7.1.8

    laravel中用到ldap认证包,要求php7.0以上版本,而且安装Mews\Captcha包的时候 验证码无法显示 报错如下: Call to undefined function Interve ...

  3. 如何自己搭建DNS服务器

    DNS服务器是计算机域名系统 (Domain Name System 或Domain Name Service) 的缩写,它是由 解析器和域名服务器组成的.域名服务器是指保存有该网络中所有主机的域名和 ...

  4. poj2987 Firing[最小割]

    题目 求选最少点个数的最大权闭合子图.(板子题) 最小割入门题,什么都不想说,丢个别人题解地址就跑. 附加几点个人理解:与s相通的S点集是闭合子图,剩下的与t相通的T点集是其他的.任意一个割都保证了有 ...

  5. C# 跨窗体事件

    跨窗体事件:例如从一个窗体改变另一个窗体button的颜色,首先需要将需要改变button的属性改为public using System; using System.Drawing; using S ...

  6. poj 3662 Telephone Lines dijkstra+二分搜索

    Telephone Lines Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 5696   Accepted: 2071 D ...

  7. 关于spark与scala版本问题记录

    记录一下版本问题: spark与scala版本对应问题: 1.官网会给出,如下,spark2.3.1默认需要scala2.11版本 2.在maven依赖网中也可以看到,如下 3.关于idea开发版本中 ...

  8. HDU2082 找单词

    问题分析 不难想到用母函数做. 令自变量\(x\)的次数就是单词价值,那么答案就是\(x\)的\(1\)次到\(50\)次的系数之和.由于我们只需要处理前\(51\)项,所以暴力多项式相乘即可. 举个 ...

  9. postman教学视频百度网盘转载分享

    百度云盘教学视频分享:https://pan.baidu.com/s/1r_e08FOkvQBZcC5-vU5M4w postman官网及下载地址:https://www.getpostman.com ...

  10. Redis大 key的发现与删除方法全解析

    个推作为国内第三方推送市场的早期进入者,专注于为开发者提供高效稳定的推送服务,经过9年的积累和发展,服务了包括新浪.滴滴在内的数十万APP.由于我们推送业务对并发量.速度要求很高,为此,我们选择了高性 ...