首頁 > 後端開發 > php教程 > PHP 迴圈中「引用」引發的奇怪問題

PHP 迴圈中「引用」引發的奇怪問題

WBOY
發布: 2016-07-25 08:46:29
原創
1014 人瀏覽過

本文整理自 stackoverflow 網站上的一篇文章 Strange behaviour after loop by reference - Is this a PHP bug? —— 在 PHP 迴圈中,如果使用 引用 會引發非常奇怪的行為 - 這是 PHP 的 bug 嗎?

問題

在我寫一個簡單的 PHP 腳本時,發生了一些非常奇怪的現象。下面是我的程式碼,為了清楚的表達我的意思,我特意去掉了一些不必要的程式碼:

  1. $arr = array("foo",
  2. "bar",
  3. "baz");
  4. "bar",
  5. "baz");
  6. foreach ($arr as &$item) { /* do nothing by reference */ }
  7. print_r($arr);
foreach ($arr as $item) { /* do nothing by value */ }
print_r($arr); // $arr has changed....why?

複製程式碼

輸出如下:
  1. Array
  2. (
  3. [0] => foo
  4. [1] => bar
  5. [2] => baz
  6. )
  7. Array
  8. (
  9. [0] => foo
  10. [1] => bar
[2] => bar // 錯誤發生??
)

複製程式碼

這是 PHP 的一個 bug 嗎? PHP 中為什麼會發生如此古怪的行為?

解析

在第一個 foreach 迴圈結束後,$item 仍然引用(reference)著數組的最後一個元素,也就是 $arr[2]。 因此,當開始第二個循環的時候,$item 變數每次循環都會被賦一個新值。 在 php 中,如果一個記憶體空間是被引用的,那麼當改變它的時候是直接改變這塊記憶體空間的值。 當改變 $item 的時候,其實也改變了 $arr[2] 的值。

因此,在第二個循環中:

第一次循環,$item 和 $arr[2] 的值變成 $arr[0],也就是 'foo'。 第二次循環,$item 和 $arr[2] 的值變成 $arr[1],也就是 'bar'。 第三次循環,$item 和$arr[2] 的值變成$arr[2],也就是'bar'($arr[2] 的值不是'baz',因為在第二次循環中變成了'bar')。 'baz' 的值實際上是在第二個循環中遺失了。 譯者

:我不喜歡把 reference 翻譯成「引用」,當然了,更不能翻譯成「參考」了。每次我像別人解釋 reference 時,都會告訴他:

reference 是 alias

。 例如你叫吳毅昌(呵呵,無異常),二狗子是你的別名。本著好兄弟好基友的情誼:“來,二狗子,這 100 塊錢給你吧。” 你——吳毅昌——回家一模口袋,多了 100 塊。 @justjavac

偵錯輸出

我們可以修改程式碼來偵錯並追蹤循環的執行細節。 我們可以輸出 $item 的值,並且遞歸的輸出陣列 $arr。
當第一個循環運作時,我們可以看到這樣的輸出:
  1. foo
  2. Array ( [0] => foo [1] => bar [2] => baz )
  3. bar
  4. Array ( [ 0] => foo [1] => bar [2] => baz )
baz
Array ( [0] => foo [1] => bar [2] => baz )

複製程式碼

在循環結束後,$item 和 $arr[2] 指向同一個記憶體區域。
當第二個循環運作時,我們看到這樣的輸出:
  1. foo
  2. Array ( [0] => foo [1] => bar [2] => foo )
  3. bar
  4. Array ( [ 0] => foo [1] => bar [2] => bar )
bar
Array ( [0] => foo [1] => bar [2] => bar )

複製程式碼

在這次循環中,需要注意隨著每次$item 被賦予一個新值, $arr[2] 也會被賦值為和$item 相同的值,因為它們都仍然指向相同的內存空間(譯註:原文寫的是$arr[3],疑為原作者筆誤。 當循環到達數組的第三個值時,它包含的值是 bar,因為它的值在前兩個循環中,被修改了。

還有疑問

也許你覺得,我只是執行了一個空迴圈 foreach ($arr as &$item){},迴圈體裡面什麼都沒做,為什麼陣列元素卻改變了?
可能你覺得這個程式碼應該等價於
  1. for ($i = 0; $i // do nothing
}

複製程式碼

其實不對,程式碼應該等價於:
  1. for ($i = 0; $i $item = $arr[$i];
}
複製程式碼

也就是說, 在 foreach 迴圈中,隱含了一個賦值運算,唯一不同的時, 在賦值過程中,我們使用了引用,所以在第一個循環中,無意中修改了正在循環的數組內部的元素。

PHP


來源:php.cn
本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn
熱門教學
更多>
最新下載
更多>
網站特效
網站源碼
網站素材
前端模板