全面解读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)

原理说明:

  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; //1

原理说明:

定义变量 $a ,并将 $a 的引用赋给变量 $b

$a = 1;
$b = &$a;

销毁 $b

unset($b);

输出 $a

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

echo $a;  //1

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

`

原文地址:https://www.cnblogs.com/chrdai/p/11061174.html

时间: 2024-11-09 02:09:30

全面解读php面试-引用变量(&)的相关文章

360大牛带你横扫PHP职场 全面解读PHP面试

第1章 课程介绍让大家了解基本面试流程和面试的核心要求以及意义是什么并理解PHP面试考点主要以基础为核心,说明PHP面试考察范围. 第2章 PHP基础知识考察点本章主要讲解技术面试时笔试考察中所遇到的PHP基础知识各个方面的考察点,帮助大家梳理PHP基础知识及易出错及难点,掌握引用变量.常量及数据类型.运算符.流程控制.自定义函数及内部函数.正则表达式.文件及目录处理.会话控制.面向对象.网络协议以及开发环境CGI.FastCGI及配置考察点内容,基本可以摸清面试官所考察... 第3章 Java

实习第二天-对象-对象引用-引用变量-精-精-精-下雨天

class Person{ } Person是一个数据类型-引用类型 数据类型-变量名   Person a;  声明一个引用类型的变量a,然后在栈中给引用变量a分配了内存空间 初学Java时,在很长一段时间里,总觉得基本概念很模糊.后来才知道,在许多Java书中,把对象和对象的引用混为一谈.可是,如果我分不清对象与对象引用, 那实在没法很好地理解下面的面向对象技术.把自己的一点认识写下来,或许能让初学Java的朋友们少走一点弯路. 为便于说明,我们先定义一个简单的类: class Vehicl

PHP字符串中引用变量问题

php中字符串引用变量主要是数据库操作字符串的不同 1,普通字符串引用变量 a.外层必须引用双引号 b.字符串中的变量可写如:$s 或者{$s} 2.数据库操作字符串(数据库操作指令) a.字符串外层必须使用双引号 b.变量必须用大括号{}(赋值字符串属性时除外) 如: `    c.当需变量与数据库属性相赋值或者作逻辑运算时,需要对应数据属性的类型:如     上述代码中数据库中的属性 title,authordid,content均为字符串,所以变量要用单引号'{$title}'或者'$ti

引用变量

引用变量是一种特殊类型的变量,将函数形参声明为此种类型的变量,形参将成为原变量的一个引用(而不是拷贝).一个引用变量的实质是另一个变量的一个别名,任何对引用变量的改变实际上都会作用到原变量上. 声明一个引用变量应在变量名前放置一个“&”.如:int &refVar;  int & refVar;  int& refVar; #include<iostream>    using namespace std; int main(){ int count = 1;

那些年我们一起追过的的&quot;引用变量&quot;--总结1

hello ,好久没来了. 今天我来总结一下有关引用变量的注意事项,一是加深一下自己的理解,二是对这块不太理解的同学可以看看. 大神可飘过,有什么不对或不足的地方请多多指教,谢谢. 假设场景: 有一个统计游戏玩家信息调查问卷系统,玩家填写了调查问卷,会给玩家一些奖励,当然目前这不是我们关注的部分. 我们需要记录一下玩家的姓名,年龄,邮箱,以及玩家曾经玩过的游戏有哪些. 既然要记录玩家玩过的游戏,必然要有Game类: package indi.bruce.summary; public class

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

final修饰的引用变量一旦初始化赋值之后就不能再指向其他的对象,那么该引用变量指向的对象的内容可变吗?看下面这个例子: public class Test { public static void main(String[] args) { final MyClass myClass = new MyClass(); System.out.println(++myClass.i); } } class MyClass { public int i = 0; } 这段代码可以顺利编译通过并且有输

JAVA 变量的定义和使用【引用变量的特殊性】

Java中主要有2类主变量: 基本数据类型:byte.short.int.long 和 float.double:或者叫主数据类型 和 引用数据类型:数组.类.接口:引用变量只会保存引用,而不是保存对象本身: 引用传递就是一段内存的使用权,一块内存可以供多人使用: 因为,Java中[数组是对象],不管里面存放的是主数据类型还是对象引用,但是可以声明出可以装在主数据类型数值的数组, 数组对象可以有主数据类型的元素,但是数组本身绝对不是主数据类型??????? 所以,Java中数组的使用,必须经过“

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

引用变量在创建的时候就必须初始化.无法创建一个未被初始化的引用. #include <iostream> using namespace std; int main() { int x=10; int y=20; int &r1; } 编译结果: 如果引用未被初始化,编译将报错. 修改引用: 引用总是指向初始化的那个变量,也就是说,引用一旦被创建并初始化之后就无法改变.这一规则有点让人迷惑.. 如果声明了一个引用的同时使用一个变量赋值了,那么这个引用就会一直指向这个变量. 在此后使用变

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

转自:http://www.cnblogs.com/huangyu/archive/2004/08/02/29622.html 从宏观的角度来看,对象是类的实例.比如: //定义一个名为Someone的类,代表这么一些人(通过指定年龄,性别,性格等基本信息)class Someone {  public int age;  public string sex;  public string name;  //other...}//......//创建一个Someone类的对象,代表昵称为nemo