搞定PHP面试 - 深入了解引用
1. 什么是引用
在 PHP 中引用是指用不同的名字访问同一个变量内容。
PHP 中的变量名和变量内容是不一样的, 因此同样的内容可以有不同的名字。
最接近的比喻是 Unix 的文件名和文件本身——变量名是目录条目,而变量内容则是文件本身。引用可以被看作是 Unix 文件系统中的硬链接。
PHP 中的引用并不像 C 的指针:例如你不能对他们做指针运算。引用并不是实际的内存地址,而是符号表别名。
2. 引用的特性
PHP 的引用允许用两个变量来指向同一个内容。
$a =& $b;
这意味着 $a 和 $b 指向了同一个变量。
$a 和 $b 在这里是完全相同的,这并不是 $a 指向了 $b 或者相反,而是 $a 和 $b 指向了同一个地方。
如果具有引用的数组被复制,其值不会解除引用。将数组传值给函数也是如此。
$a = 'a';
$arr1 = [
'a' => $a,
'b' => &$a, // $arr1['b'] 与 $a 指向同一个变量
];
// 将 $arr1 传值赋值给 $arr2
$arr2 = $arr1;
print_r($arr2); // $arr2 的值为 ['a' => 'a', 'b' => 'a']
// 修改 $a 的值为 'b'
$a = 'b';
print_r($arr2); // $arr2 的值为 ['a' => 'a', 'b' => 'b']
function foo($arr){
// 将 $arr['b'] 的值改为 'c';
$arr['b'] = 'c';
}
echo $a; // $a 的值为 'b'
// 将 $arr1 传入函数
foo($arr1);
echo $a; // $a 的值为 'c'
如果对一个未定义的变量进行引用赋值、引用参数传递或引用返回,则会自动创建该变量。
// 定义函数 foo(),通过引用传递参数
function foo(&$var) { }
foo($a); // 创建变量 $a,值为 NULL
var_dump($a); // NULL
foo($b['b']); // 创建数组 $b = ['b' => NULL]
var_dump(array_key_exists('b', $b)); // bool(true)
$c = new StdClass;
foo($c->d); // 创建对象属性 $c->d = NULL
var_dump(property_exists($c, 'd')); // bool(true)
如果在一个函数内部给一个声明为 global 的变量赋于一个引用,该引用只在函数内部可见。可以通过使用 $GLOBALS 数组避免这一点。
$var1 = 'var1';
$var2 = 'var2';
function global_references($use_globals)
{
global $var1, $var2;
if (!$use_globals) {
$var2 = & $var1; // $var2 只在函数内部可见
} else {
$GLOBALS["var2"] = & $var1; // $GLOBALS["var2"]在全球范围内也可见
}
}
global_references(false);
echo "var2 is set to '$var2'\n"; // var2 is set to 'var2'
global_references(true);
echo "var2 is set to '$var2'\n"; // var2 is set to 'var1'
可以把 global $var; 当成是 $var =& $GLOBALS['var']; 的简写。从而将其它引用赋给 $var 只改变了本地变量的引用。
在 foreach 语句中给一个具有引用的变量赋值,被引用的对象也被改变。
$ref = 0;
$row = & $ref;
foreach ([1, 2, 3] as $row) {
// do something
}
echo $ref; // 3 - 遍历数组的最后一个元素
3. 引用传递
可以将一个变量通过引用传递给函数,这样该函数就可以修改其参数的值。
function foo(&$var)
{
$var++;
}
$a=5;
foo($a);
echo $a; // 6
注意在函数调用时没有引用符号——只有函数定义中有。光是函数定义就足够使参数通过引用来正确传递了。
可以通过引用传递的内容:
- 变量
- 从函数中返回的引用
通过引用传递变量
function foo(&$var)
{
$var++;
}
$a=5;
foo($a);
echo $a; // 6
通过引用传递从函数中返回的引用
function foo(&$var)
{
$var++;
echo $var; // 6
}
function &bar()
{
$a = 5;
return $a;
}
foo(bar());
不能通过引用传递函数、表达式、值等
function foo(&$var)
{
$var++;
}
function bar() // 注意,这个函数不返回引用
{
$a = 5;
return $a;
}
foo(bar()); // 自 PHP 5.0.5 起导致致命错误,自 PHP 5.1.1 起导致严格模式错误,自 PHP 7.0 起导致 notice 信息
foo($a = 5); // 表达式,不是变量。PHP Notice: Only variables should be passed by reference
foo(5); // PHP Fatal error: Only variables can be passed by reference
4. 引用返回
当你想要使用一个函数来找到一个引用应该被绑定的变量时,可以使用引用返回。
不要用返回引用来增加性能,引擎足够聪明,可以自己进行优化。仅在有合理的技术原因时才返回引用!
class Foo {
public $value = 42;
public function &getValue() {
return $this->value;
}
}
$foo = new Foo;
// $myValue 是 $obj->value 的引用.
$myValue = &$foo->getValue();
// 将 $foo->value 修改为 2
$foo->value = 2;
echo $myValue; // 2
与参数引用传递不同,引用返回必须在两个地方都用 & 符号 —— 指出返回的是一个引用,而不是通常的一个拷贝,同样也指出 $myValue 是作为引用的绑定,而不是通常的赋值。
引用返回只能返回变量。如果试图这样从函数返回引用:return intval($this->value);,将会报错,因为函数在试图返回一个表达式的结果而不是一个引用的变量。只能从函数返回引用变量——没别的方法。
class Foo {
public $value = 42;
public function &getValue() {
return intval($this->value); // PHP Notice: Only variable references should be returned by reference
}
}
$foo = new Foo;
// $myValue 是 $obj->value 的引用.
$myValue = &$foo->getValue();
5. 取消引用
当 unset 一个引用,只是断开了变量名和变量内容之间的绑定。这并不意味着变量内容被销毁了。
$a = 1;
$b = & $a;
unset($a);
echo $b; // 1
6. 发现
许多 PHP 的语法结构是通过引用机制实现的,所以上述有关引用绑定的一切也都适用于这些结构。
global 引用
当用 global $var 声明一个变量时实际上是在函数内部建立了一个到全局变量的引用。也就是说这样做的效果是相同的:
global $var;
$var =& $GLOBALS["var"];
这意味着,unset $var 不会 unset 掉全局变量 $GLOBALS["var"]。
$this
在一个对象的方法中,$this 永远是调用它的对象的引用。
原文地址:https://segmentfault.com/a/1190000016373107
搞定PHP面试 - 深入了解引用的更多相关文章
- [算法总结] 13 道题搞定 BAT 面试——字符串
1. KMP 算法 谈到字符串问题,不得不提的就是 KMP 算法,它是用来解决字符串查找的问题,可以在一个字符串(S)中查找一个子串(W)出现的位置.KMP 算法把字符匹配的时间复杂度缩小到 O(m+ ...
- 【搞定Jvm面试】 Java 内存区域揭秘附常见面试题解析
本文已经收录自笔者开源的 JavaGuide: https://github.com/Snailclimb ([Java学习+面试指南] 一份涵盖大部分Java程序员所需要掌握的核心知识)如果觉得不错 ...
- 搞定PHP面试 - 变量知识点整理
一.变量的定义 1. 变量的命名规则 变量名可以包含字母.数字.下划线,不能以数字开头. $Var_1 = 'foo'; // 合法 $var1 = 'foo'; // 合法 $_var1 = 'fo ...
- 【搞定Jvm面试】 JVM 垃圾回收揭秘附常见面试题解析
JVM 垃圾回收 写在前面 本节常见面试题 问题答案在文中都有提到 如何判断对象是否死亡(两种方法). 简单的介绍一下强引用.软引用.弱引用.虚引用(虚引用与软引用和弱引用的区别.使用软引用能带来的好 ...
- 30秒搞定String面试
Java 语言中,无论新菜鸟,还是老司机,真正了解String内存的很少.关于String 的试题,花样很多.== 在什么情况下是true,什么情况是false.我总结出如下3点让你彻底结束对Stri ...
- 搞定PHP面试 - 正则表达式知识点整理
一.简介 1. 什么是正则表达式 正则表达式(Regular Expression)就是用某种模式去匹配一类字符串的一种公式.正则表达式使用单个字符串来描述.匹配一系列匹配某个句法规则的字符串.正则表 ...
- 搞定PHP面试 - 运算符知识点整理
一.算术运算符 1. 概览 例子 名称 结果 $a + $b 加法 $a 和 $b 的和. $a - $b 减法 $a 和 $b 的差. $a * $b 乘法 $a 和 $b 的积. $a / $b ...
- 搞定PHP面试 - 函数知识点整理
一.函数的定义 1. 函数的命名规则 函数名可以包含字母.数字.下划线,不能以数字开头. function Func_1(){ } //合法 function func1(){ } //合法 func ...
- 【搞定Jvm面试】 JDK监控和故障处理工具揭秘
本文已经收录自笔者开源的 JavaGuide: https://github.com/Snailclimb ([Java学习+面试指南] 一份涵盖大部分Java程序员所需要掌握的核心知识)如果觉得不错 ...
随机推荐
- 软件project师周兆熊给IT学子的倾情奉献
[来信] 贺老师: 你好,我是中兴通讯的一名软件开发project师,名叫周兆熊. 近期看了您的新书<逆袭大学:传给IT学子的正能量>,感觉你真心为当代学子答疑解惑.非常值得敬佩! 从上大 ...
- Oracle创建自增字段方法-ORACLE SEQUENCE的简介
曾经最头疼的就是对表插入数据的时候,有主键问题. 由于主键不可以反复,所以得用函数自己定义一个规则生成不反复的值赋值给主键. 如今发现oracle有sequence就不用那么麻烦了. 转自:http: ...
- Android ListView拉到顶/底部,像橡皮筋一样弹性回弹复位
<Android ListView拉到顶/底部,像橡皮筋一样弹性回弹复位> Android本身的ListView拉到顶部或者底部会在顶部/底部边缘间隙出现一道"闪光&quo ...
- Android端 配置极光推送
由于业务须要,androidclient须要加推送.原来採用的百度推送.可是小米手机有时候收不到.后来换成了极光推送,极光的话全部设备都能收到推送,可是在高峰的时候会推迟.博主说的免费版的,收费的没用 ...
- Gridview表格控件
Gridview表格控件 效果图: 分析: 使用和ListvVew很像,都是经过适配器将数据绑定到控件上 具体步骤如下: 1.创建GridView控件,并指定列数 android:numColumns ...
- 南海区行政审批管理系统接口规范v0.3(规划)
1. 会话API 1.1. login [登录验证] {"r_code":"500","r_msg":"操作失败",&q ...
- 自己实现的一个 .net 缓存类(原创)
public class CacheContainer { private static Hashtable ht = new Hashtable(); /// <summary> /// ...
- Hdu-6249 2017CCPC-Final G.Alice’s Stamps 动态规划
题面 题意:给你n个集合,每个集合有L到R这些种类的邮票,让你选择其中的K个集合,使得最后选择的邮票种类尽可能多,N,L,R都<=2000 题解:容易乱想到网络流,可是再细想一下就会发现处理不了 ...
- BPM不同表单之间子表的赋值
上次写的是同一个表单的子表之间赋值,这次是不同表单之间子表的赋值 首先,我们给需要赋值的表单添加一个复制按钮 $.MvcSheet.AddAction({ Action: &qu ...
- 在redhat6.4上编译z3求解器
因为项目需要,我们使用到了微软的z3求解器求约束,但是z3求解器在红帽平台上并没有发布编译好的二进制版本,而我们的运行环境是红帽的企业版6.4,因此需要自己编译相应的二进制. z3是由微软公司开发的一 ...