
本文深入探讨了php `foreach` 循环中尝试通过引用重新赋值数组元素的常见误区。当在 `foreach ($arr as &$vl)` 中执行 `$vl = &$anothervar;` 时,`$vl` 的引用目标会改变,但数组原始元素不会随之成为新变量的引用。文章将解释这一行为,并提供使数组元素引用外部变量的正确方法。
在PHP开发中,我们经常需要对数组元素进行操作,有时甚至希望将数组元素设置为引用某个外部变量。然而,在 foreach 循环中使用引用时,如果不理解其底层机制,很容易遇到意想不到的行为。
考虑以下代码示例,其中我们试图让数组 $arr 的所有元素都引用外部变量 $val:
'AAA', 'b' => 'BBB']; echo "初始数组: " . print_r($arr, true) . "
"; // 预期输出: Array ( [a] => AAA [b] => BBB ) // 方法1: 直接赋值引用 - 有效 $arr['a'] = &$val; $arr['b'] = &$val; echo "方法1 (直接赋值引用) 后: " . print_r($arr, true) . "
"; // 预期输出: Array ( [a] => OOOOOO [b] => OOOOOO ) // 因为 $arr['a'] 和 $arr['b'] 现在都引用 $val,当 $val 改变时,它们也会改变。 // 重置数组,用于演示方法2 $arr = ['a' => 'AAA', 'b' => 'BBB']; echo "重置数组: " . print_r($arr, true) . "
"; // 方法2: 在 foreach 循环中尝试重新赋值引用 - 无效 foreach ($arr as $ky => &$vl) { // 此时,$vl 是 $arr[$ky] 的一个引用(别名) // 这一行代码尝试将 $vl 重新赋值为 $val 的引用 $vl = &$val; } echo "方法2 (foreach 中重新赋值引用) 后: " . print_r($arr, true) . "
"; // 实际输出: Array ( [a] => AAA [b] => BBB ) // 数组元素并未引用 $val,保持原样。 ?>
从上述示例中可以看到,“方法1”成功地使数组元素引用了 $val,而“方法2”却失败了。这背后的原因在于PHP中引用和 foreach 循环的交互方式。
理解 foreach 循环中的引用行为
当使用 foreach ($arr as $key => &$value) 语法时,$value 变量会成为当前迭代数组元素的一个引用(或别名)。这意味着,对 $value 的任何修改都会直接反映到原始数组 $arr[$key] 中。
立即学习“PHP免费学习笔记(深入)”;
例如,如果我们想通过 foreach 循环修改数组元素的 值:
2 [1] => 4 [2] => 6 ) ?>
这种情况下,$num 作为 $arr_values 中元素的引用,对其赋值操作会直接修改原数组元素的值,这是符合预期的。
foreach 中重新赋值引用的陷阱
然而,当我们尝试在 foreach 循环中执行 $vl = &$val; 这样的操作时,情况就变得不同了。
- 初始状态: 在每次循环迭代开始时,$vl 确实是当前数组元素(例如 $arr['a'])的一个引用。这意味着 $vl 和 $arr['a'] 指向内存中的同一个位置。
- 重新赋值引用: 当执行 $vl = &$val; 时,你并不是在修改 $arr['a'] 的引用目标。相反,你是在告诉PHP,现在 $vl 这个变量应该 停止引用 $arr['a'],转而 引用 $val。
- 影响范围: 这种操作只改变了 $vl 自身的引用目标,而没有改变 $arr['a'] 的引用目标。$arr['a'] 仍然保持其原有的状态(要么是原始值,要么是它之前引用的某个变量)。一旦当前循环迭代结束,$vl 变量就会超出作用域(或者在下一次迭代中重新绑定到下一个数组元素),它对 $val 的引用关系也随之消失,对原始数组 $arr 没有任何持久影响。
简而言之,$vl = &$val; 改变的是 $vl 这个局部变量的“指向”,而不是它所指向的那个原始数组元素的“指向”。
正确的解决方案
如果目标是让数组的每个元素都引用一个外部变量,那么必须直接操作数组元素本身,而不是通过 foreach 循环中的引用变量来间接尝试重新绑定。
以下是实现这一目标的两种有效方法:
1. 直接通过键名赋值引用
这是最直接且清晰的方法,也是“方法1”所采用的策略。
'AAA', 'b' => 'BBB'];
// 遍历数组,直接通过键名将数组元素设置为 $val 的引用
foreach ($arr as $ky => $value) { // 注意这里 $value 不再是引用,因为我们直接操作 $arr[$ky]
$arr[$ky] = &$val;
}
echo "正确方法 (直接通过键名赋值引用) 后: " . print_r($arr, true) . "
";
// 输出: Array ( [a] => OOOOOO [b] => OOOOOO )
// 验证引用关系
$val = 'NEW_VALUE';
echo "修改 $val 后: " . print_r($arr, true) . "
";
// 输出: Array ( [a] => NEW_VALUE [b] => NEW_VALUE )
?>这种方法明确地将 $arr[$ky] 设置为 $val 的引用,从而实现了预期的效果。
2. 如果仅需修改值而非引用目标
如果你的目的仅仅是修改数组元素的值,而不是让它们引用另一个变量,那么 foreach ($arr as &$vl) 语法是完全有效的。
1, 'two' => 2];
foreach ($arr_modify_values as &$item) {
$item += 10; // 修改 $arr_modify_values['one'] 和 $arr_modify_values['two'] 的值
}
echo "修改值后的数组: " . print_r($arr_modify_values, true) . "
";
// 输出: Array ( [one] => 11 [two] => 12 )
?>这再次强调了 foreach 引用用于修改 值 的有效性,但不能用于改变原始数组元素本身的 引用目标。
总结
在PHP中处理引用时,理解变量的“引用目标”和“变量本身”是至关重要的。当在 foreach ($arr as $ky => &$vl) 循环中,$vl 是一个指向 $arr[$ky] 的引用。如果你尝试执行 $vl = &$anotherVar;,你只是改变了 $vl 这个局部变量的引用目标,而没有改变 $arr[$ky] 的引用目标。要使数组元素引用一个外部变量,必须直接通过 $arr[$ky] = &$anotherVar; 的方式进行赋值。掌握这一机制可以避免在PHP引用编程中常见的陷阱,确保代码行为符合预期。










