最近遇到這樣的問題
$data = [1,2,3];
foreach ($data as &$v){}
foreach ($data as $v){}
print_r($data); //這里輸出內容是什么 輸出[1,2,2]
第一個foreach
首先理解&,它指向變量的內存地址,可以理解為C語言中的指針。ok ,先看看第一個foreach后,$data的內容
array(3) {
[0]=>
int(1)
[1]=>
int(2)
[2]=>
&int(3)
}
從上面可以看出$data[2]類型&int(3),變成了存儲$data[2]的內存指針,由于foreach 結束之后$v并不會馬上注銷掉的,因此第一次foreach后$v也指向了data[2]的引用
$v = &$data[2]
也就說第一個foreach結束后,$v轉變成了內存地址的引用不是簡單的變量賦值了。
第二個foreach
foreach ($data as $v){}
//遍歷第一個元素
$v = $data[0]; //此時$data[0]的值賦給 $v,而$v指向$data[2]的內存地址會導致$data[2] = $data[0],那么此時的$data為[1,2,1]
//遍歷第二個元素
$v = data[1];//同理,此時$data變成了[1,2,2]
//遍歷第三個元素
$v = data[2];//同理此時$data變成了[1,2,2]
嗯嗯,看以下代碼
$a['a'] = 123;
$b = &$a['a']; //此時的$a['a']為引用類型了
$c = $a;//c['a'] = a['a']也為引用類型
$c['a'] = 456;
print_r($a);//輸出456
再看以下代碼
$a['a'] = 123;
$b = &$a;
$c = $a;//純粹的賦值
$c['a'] = 456;
print_r($a);//輸出123
思考為什么第一段代碼輸出456,第二段代碼輸出123?
答:$b = &$a['a'];與$b = &$a;的區別,前者把數組a的元素變成的引用類型,后者是把a變量變成了引用類型, 所以當執行到$c = $a;賦值運算時,前者的$b會包含引用類型,而后者則是正常的賦值。
總結
在foreach 循環中慎用& ,容易出現意想不到的bug
修改引用類型變量的值,會改變該引用變量所指向的內容
參考資料
https://learnku.com/articles/7001/php-ray-foreach-and-references-thunder
https://www.laruence.com/2018/04/08/3179.html