搞定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 永遠是呼叫它的物件的引用。