php 引用详解
1 什么是引用?
在 PHP 中引用意味着用不同的名字访问同一个变量内容。这并不像 C 的指针:例如你不能对他们做指针运算,他们并不是实际的内存地址...... 查看引用不是什么了解更多信息。 替代的是,引用是符号表别名。
注意在PHP 中,变量名和变量内容是不一样的, 因此同样的内容可以有不同的名字。最接近的比喻是 Unix 的文件名和文件本身——变量名是目录条目,而变量内容则是文件本身。引用可以被看作是 Unix 文件系统中的硬链接。
当作为赋值的变量处理时,引用的行为与预期一致。然而,它们实际上是引用原始数据的对象
$var = "foo"; $ref1 =& $var; // new object that references $var $ref2 =& $ref1; // references $var directly, not $ref1!!!!! echo $ref2; // >foo unset($ref1); echo $ref1; // >Notice: Undefined variable: ref1 echo $ref2; // >foo echo $var; // >foo
2 引用特性
PHP 的引用允许用两个变量来指向同一个内容。意思是,当这样做时
$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'
如果对一个未定义的变量进行引用赋值、引用参数传递或引用返回,则会自动创建该变量。
function foo(&$var) { } foo($a); // $a is "created" and assigned to null $b = array(); foo($b['b']); var_dump(array_key_exists('b', $b)); // bool(true) $c = new StdClass; foo($c->d); var_dump(property_exists($c, 'd')); // bool(true)
同样的语法可以用在函数中,它返回引用,以及用在 new 运算符中(PHP 4.0.4 以及以后版本):
$bar =& new fooclass(); $foo =& find_var($bar);
自 PHP 5 起,new 自动返回引用,因此在此使用 =& 已经过时了并且会产生 E_STRICT 级别的消息。
如果在一个函数内部给一个声明为 global 的变量赋于一个引用,该引用只在函数内部可见。可以通过使用 $GLOBALS 数组避免这一点。
$var1 = "Example variable"; $var2 = ""; function global_references($use_globals) { global $var1, $var2; if (!$use_globals) { $var2 =& $var1; // visible only inside the function } else { $GLOBALS["var2"] =& $var1; // visible also in global context } } global_references(false); echo "var2 is set to '$var2'\n"; // var2 is set to '' global_references(true); echo "var2 is set to '$var2'\n"; // var2 is set to 'Example variable'
把 global $var; 当成是 $var =& $GLOBALS['var']; 的简写。从而将其它引用赋给 $var 只改变了本地变量的引用。
如果在 foreach 语句中给一个具有引用的变量赋值,被引用的对象也被改变。
$ref = 0; $row =& $ref; foreach (array(1, 2, 3) as $row) { // do something } echo $ref; // 3 - last element of the iterated array
引用做的第二件事是用引用传递变量。这是通过在函数内建立一个本地变量并且该变量在呼叫范围内引用了同一个内容来实现的。例如:
function foo(&$var) { $var++; } $a=5; foo($a);
将使 $a 变成 6。这是因为在 foo 函数中变量 $var 指向了和 $a 指向的同一个内容。
3 引用传递
可以将一个变量通过引用传递给函数,这样该函数就可以修改其参数的值(简称:引用传递参数)。语法如下:
function foo(&$var) { $var++; } $a=5; foo($a); // $a is 6 here
↑↑↑注意在函数调用时没有引用符号——只有函数定义中有。光是函数定义就足够使参数通过引用来正确传递了。
function &bar() { $a = 5; return $a; } foo(bar());
任何其它表达式都不能通过引用传递,结果未定义。只能是引用传递变量。
function foo(&$var) { $var++; } function bar() // Note the missing & { $a = 5; return $a; } foo(bar()); // 自 PHP 5.0.5 起导致致命错误,自 PHP 5.1.1 起导致严格模式错误 // 自 PHP 7.0 起导致 notice 信息 foo($a = 5) // 表达式,不是变量 foo(5) // 导致致命错误
4 引用返回
引用返回用在当想用函数找到引用应该被绑定在哪一个变量上面时。不要用返回引用来增加性能,引擎足够聪明来自己进行优化。仅在有合理的技术原因时才返回引用!要返回引用,使用此语法:
class foo { public $value = 42; public function &getValue() { return $this->value; } } $obj = new foo; $myValue = &$obj->getValue(); // $myValue is a reference to $obj->value, which is 42. $obj->value = 2; echo $myValue; // prints the new value of $obj->value, i.e. 2.
本例中 getValue 函数所返回的对象的属性将被赋值,而不是拷贝,就和没有用引用语法一样。
注意:和参数传递不同,这里必须在两个地方都用 & 符号——指出返回的是一个引用,而不是通常的一个拷贝,同样也指出 $myValue 是作为引用的绑定,而不是通常的赋值。
注意:如果试图这样从函数返回引用:return ($this->value);,这将不会起作用,因为在试图返回一个表达式的结果而不是一个引用的变量。只能从函数返回引用变量——没别的方法。如果代码试图返回一个动态表达式或 new 运算符的结果,自 PHP 4.4.0 和 PHP 5.1.0 起会发出一条E_NOTICE
错误。
5 取消引用
当 unset 一个引用,只是断开了变量名和变量内容之间的绑定。这并不意味着变量内容被销毁了。例如:
$a = 1; $b =& $a; unset($a);