资讯专栏INFORMATION COLUMN

搞定PHP面试 - 变量的引用赋值与传值赋值

liangdas / 2562人阅读

摘要:比如变量的多次赋值函数参数传递,并在函数体内修改实参等。引用赋值定义一个变量定义变量,将变量的引用赋给对进行修改输出结果定义一个变量定义变量,将变量的引用赋给对进行修改二使用查看变量的引用情况用于显示变量的信息。

一、使用 memory_get_usage() 查看PHP内存使用量 1. 传值赋值
// 定义一个变量
$a = range(0, 10000);
var_dump(memory_get_usage());

// 定义变量b,将a变量的值赋值给b
$b = $a;
var_dump(memory_get_usage());

// 对a进行修改
// COW: Copy-On-Write
$a = range(0, 10000);
var_dump(memory_get_usage());

输出结果:

int(989768)
int(989856)
int(1855608)
定义一个变量 $a = range(0, 10000);

$b = $a;

对a进行修改 $a = range(0, 10000);

PHP写时复制机制(Copy-on-Write,也缩写为COW)

顾名思义,就是在写入时才真正复制一份内存进行修改。
COW最早应用在Unix系统中对线程与内存使用的优化,后面广泛的被使用在各种编程语言中,如C++的STL等。
在PHP内核中,COW也是主要的内存优化手段。
在通过变量赋值的方式赋值给变量时,不会申请新内存来存放新变量的值,而是简单的通过一个计数器来共用内存。只有在其中的一个引用指向变量的值发生变化时,才申请新空间来保存值内容,以减少对内存的占用。
在很多场景下PHP都使用COW进行内存的优化。比如:变量的多次赋值、函数参数传递,并在函数体内修改实参等。

2. 引用赋值
// 定义一个变量
$a = range(0, 10000);
var_dump(memory_get_usage());

// 定义变量b,将a变量的引用赋给b
$b = &$a;
var_dump(memory_get_usage());

// 对a进行修改
$a = range(0, 10000);
var_dump(memory_get_usage());

输出结果:

int(989760)
int(989848)
int(989840)
定义一个变量 $a = range(0, 10000);

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

对a进行修改 $a = range(0, 10000);

二、使用 xdebug_debug_zval() 查看变量的引用情况
xdebug_debug_zval() 用于显示变量的信息。需要安装xdebug扩展。
1. 传值赋值
$a = 1;
xdebug_debug_zval("a");

// 定义变量b,把a的值赋值给b
$b = $a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

// a进行写操作
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

输出结果:

a: (refcount=1, is_ref=0)=1
a: (refcount=2, is_ref=0)=1
b: (refcount=2, is_ref=0)=1
a: (refcount=1, is_ref=0)=2
b: (refcount=1, is_ref=0)=1
定义变量 $a = 1;
$a = 1;
xdebug_debug_zval("a");

输出

a: (refcount=1, is_ref=0)=1

refcount=1 表示该变量指向的内存地址的引用个数变为1
is_ref=0 表示该变量不是引用

定义变量 $b ,把 $a 的值赋给 $b$b = $a;
$b = $a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

输出

a: (refcount=2, is_ref=0)=1
b: (refcount=2, is_ref=0)=1

refcount=2 表示该变量指向的内存地址的引用个数变为2
is_ref=0 表示该变量不是引用

对变量 $a 进行写操作 $a = 2;
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

输出

a: (refcount=1, is_ref=0)=2
b: (refcount=1, is_ref=0)=1

因为COW机制,对变量 $a 进行写操作时,会为变量 $a 新分配一块内存空间,用于存储变量 $a 的值。
此时 $a$b 指向的内存地址的引用个数都变为1。

2. 引用赋值
$a = 1;
xdebug_debug_zval("a");

// 定义变量b,把a的引用赋给b
$b = &$a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

// a进行写操作
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");
a: (refcount=1, is_ref=0)=1
a: (refcount=2, is_ref=1)=1
b: (refcount=2, is_ref=1)=1
a: (refcount=2, is_ref=1)=2
b: (refcount=2, is_ref=1)=2
定义变量 $a = 1;
$a = 1;
xdebug_debug_zval("a");

输出

a: (refcount=1, is_ref=0)=1

refcount=1 表示该变量指向的内存地址的引用个数变为1
is_ref=0 表示该变量不是引用

定义变量 $b ,把 $a 的引用赋给 $b$b = &$a;
$b = &$a;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

输出

a: (refcount=2, is_ref=1)=1
b: (refcount=2, is_ref=1)=1

refcount=2 表示该变量指向的内存地址的引用个数变为2
is_ref=1 表示该变量是引用

对变量 $a 进行写操作 $a = 2;
$a = 2;
xdebug_debug_zval("a");
xdebug_debug_zval("b");

输出

a: (refcount=2, is_ref=1)=2
b: (refcount=2, is_ref=1)=2

因为变量 $a 和变量 $b 指向相同的内存地址,其实引用。
对变量 $a 进行写操作时,会直接修改指向的内存空间的值,因此变量 $b 的值会跟着一起改变。

三、当变量时引用时,unset()只会取消引用,不会销毁内存空间
$a = 1;
$b = &$a;

// unset 只会取消引用,不会销毁内存空间
unset($b);

echo $a;

输出

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

销毁 $b
unset($b);

输出 $a

虽然销毁的 $b,但是 $a 的引用和内存空间依旧存在。

echo $a;

输出

1
四、php中对象本身就是引用赋值
class Person
{
    public $age = 1;
}

$p1 = new Person;
xdebug_debug_zval("p1");

$p2 = $p1;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");

$p2->age = 2;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");
p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
实例化对象 $p1 = new Person;
$p1 = new Person;
xdebug_debug_zval("p1");

输出

p1: (refcount=1, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }

refcount=1 表示该变量指向的内存地址的引用个数变为1
is_ref=0 表示该变量不是引用

$p1 赋给 $p2
$p2 = $p1;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");

输出

p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=2, is_ref=0)=1 }

refcount=2 表示该变量指向的内存地址的引用个数变为2

$p2 中的属性 age 进行写操作
$p2->age = 2;
xdebug_debug_zval("p1");
xdebug_debug_zval("p2");

输出

p1: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }
p2: (refcount=2, is_ref=0)=class Person { public $age = (refcount=1, is_ref=0)=2 }

因为php中对象本身就是引用赋值。对 $p2 中的属性 age 进行写操作时,会直接修改指向的内存空间的值,因此变量 $p1age 属性的值会跟着一起改变。

五、实战例题分析
/**
 * 写出如下程序的输出结果
 *
 * $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] 在内存中分别开辟了一块空间](http://md.ws65535.top/xsj/201...

$v = &$d[0] 改变了 $v 指向的内存地址
$v = &$d[0]

![$v = &$d[0] 改变了 $val 指向的内存地址](http://md.ws65535.top/xsj/201...

第一次循环后 $d 的值:
["a", "b", "c"]
2. 第二次循环 进入 foreach$v 被赋值为 "b",此时$v指向的内存地址与 $d[0] 相同,且为引用,因此 $d[0] 的值被修改为 "b"

$v = "b" => $d[0] = "b"

![$v = ‘b’ => $d[0] = ‘b’](http://md.ws65535.top/xsj/201...

推算出进入 foreach$d[$k] 的值
$k = 1
$d[$k] = $d[1] = "b"

![$d[2] = ‘b’](http://md.ws65535.top/xsj/201...

$v = &$d[1] 改变了 $v 指向的内存地址
$v = &$d[1]

![$v = &$d[1]](http://md.ws65535.top/xsj/201...

第二次循环后 $d 的值
["b", "b", "c"]
3. 第三次循环 进入 foreach$v 被赋值为 "c",此时$v指向的内存地址与 $d[1] 相同,且为引用,因此 $d[1] 的值被修改为 "c"

$v = "c" => $d[1] = "c"

![$v = ‘c’ => $d[1] = ‘c’](http://md.ws65535.top/xsj/201...

推算出进入 foreach$d[$k] 的值
$k = 2
$d[2] = "c"

![$d[2] = ‘c’](http://md.ws65535.top/xsj/201...

$v = &$d[2] 改变了 $v 指向的内存地址
$v = &$d[2]

![$v = &$d[2]](http://md.ws65535.top/xsj/201...

第三次循环后 $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
)

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/29360.html

相关文章

  • 搞定PHP面试 - 变量知识点整理

    摘要:声明静态变量时不能用表达式的结果对其赋值正确错误使用表达式的结果赋值错误使用表达式的结果赋值静态变量与递归函数静态变量提供了一种处理递归函数的方法。 一、变量的定义 1. 变量的命名规则 变量名可以包含字母、数字、下划线,不能以数字开头。 $Var_1 = foo; // 合法 $var1 = foo; // 合法 $_var1 = foo; // 合法 $Var-1 = foo; /...

    Mertens 评论0 收藏0
  • 搞定PHP面试 - 深入了解引用

    摘要:引用可以被看作是文件系统中的硬链接。如果具有引用的数组被复制,其值不会解除引用。如果试图这样从函数返回引用,将会报错,因为函数在试图返回一个表达式的结果而不是一个引用的变量。这并不意味着变量内容被销毁了。 1. 什么是引用 在 PHP 中引用是指用不同的名字访问同一个变量内容。PHP 中的变量名和变量内容是不一样的, 因此同样的内容可以有不同的名字。最接近的比喻是 Unix 的文件名和...

    fox_soyoung 评论0 收藏0
  • 一篇文章理解JS数据类型、深拷贝和浅拷贝

    摘要:接下来我们进入正片数据类型六种基本数据类型布尔值,和一个表明值的特殊关键字。一种数据类型,它的实例是唯一且不可改变的。在中是没有方法是可以改变布尔值和数字的。参考资料深拷贝浅拷贝 前言 笔者最近整理了一些前端技术文章,如果有兴趣可以参考这里:muwoo blogs。接下来我们进入正片: js 数据类型 六种 基本数据类型: Boolean. 布尔值,true 和 false. nu...

    EddieChan 评论0 收藏0
  • 一篇文章理解JS数据类型、深拷贝和浅拷贝

    摘要:接下来我们进入正片数据类型六种基本数据类型布尔值,和一个表明值的特殊关键字。一种数据类型,它的实例是唯一且不可改变的。在中是没有方法是可以改变布尔值和数字的。参考资料深拷贝浅拷贝 前言 笔者最近整理了一些前端技术文章,如果有兴趣可以参考这里:muwoo blogs。接下来我们进入正片: js 数据类型 六种 基本数据类型: Boolean. 布尔值,true 和 false. nu...

    enda 评论0 收藏0

发表评论

0条评论

liangdas

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<