
本文深入解析php `foreach` 循环中引用赋值的机制。在 `foreach ($arr as &$vl)` 内部,直接将 `$vl` 重新赋值为新引用 (`$vl = &$new_var;`) 不会使原数组元素 `$arr[$ky]` 也指向新引用。这是因为 `$vl` 的绑定被改变,而非 `$arr[$ky]`。文章将提供正确的方法,即通过 `$arr[$ky] = &$new_var;` 直接操作数组元素,以实现预期的引用绑定效果,避免常见陷阱。
理解PHP中的引用
在PHP中,引用(reference)允许两个变量指向同一个底层数据。这意味着对其中一个变量的修改会影响到另一个变量。引用并非指针,它更像是一个别名。当使用 & 符号进行赋值时,例如 $b = &$a;,$b 成为 $a 的一个别名,它们共享相同的值。
foreach 循环中的引用行为
foreach 循环提供了一种遍历数组或对象的方式。当我们在 foreach 循环中使用引用语法 foreach ($array as $key => &$value) 时,$value 变量会成为当前迭代到的数组元素的引用。这意味着,在循环体内对 $value 的修改会直接影响到原始数组中的对应元素。
考虑以下示例代码,其中 $val 是一个字符串,$arr 是一个关联数组:
$val = 'OOOOOO'; $arr = ['a' => 'AAA', 'b' => 'BBB']; echo "初始数组: " . print_r($arr, true) . "
"; // Array ( [a] => AAA [b] => BBB )
我们希望将 $arr 中的所有元素都变为对 $val 的引用,这样当 $val 改变时,$arr 中的元素也会随之改变。
立即学习“PHP免费学习笔记(深入)”;
方法一:直接通过键名赋值引用 (有效)
最直接的方法是使用数组的键名直接为元素赋值引用:
// 方法一:直接通过键名赋值引用 $arr['a'] = &$val; $arr['b'] = &$val; echo "方法一后数组: " . print_r($arr, true) . "
"; // Array ( [a] => OOOOOO [b] => OOOOOO )
这种方法简单明了,直接将 $arr['a'] 和 $arr['b'] 绑定为 $val 的引用。
方法二:在 foreach 循环中尝试重新绑定引用 (无效)
现在我们来看一个常见的误区。许多开发者可能尝试在 foreach 循环中使用引用别名 $vl 来重新绑定引用:
// 方法二:在 foreach 循环中尝试重新绑定引用 (无效)
// $arr = ['a' => 'AAA', 'b' => 'BBB']; // 重置数组以便测试
// foreach ($arr as $ky => &$vl) {
// // 期望:$arr[$ky] 变为 $val 的引用
// $vl = &$val;
// }
// echo "方法二后数组: " . print_r($arr, true) . "
"; // 实际输出:Array ( [a] => AAA [b] => BBB )这段代码执行后,$arr 数组的元素并不会变成对 $val 的引用。为什么会这样?
当 foreach ($arr as $ky => &$vl) 循环开始时:
- 在第一次迭代中,$vl 成为 $arr['a'] 的引用。这意味着 $vl 和 $arr['a'] 指向同一块内存。
- 接着执行 $vl = &$val;。这个操作的含义是“将变量 $vl 重新绑定为 $val 的引用”。此时,$vl 不再是 $arr['a'] 的引用,而是变成了 $val 的引用。
- 关键在于:这个重新绑定操作只影响了 $vl 本身,而没有改变 $arr['a'] 的引用目标。 $arr['a'] 仍然保持其原始状态(即一个值为 'AAA' 的普通变量)。
- 当循环进入下一次迭代时,$vl 会再次被重新绑定为 $arr['b'] 的引用,然后再次被重新绑定为 $val 的引用,如此往复。
因此,在循环结束后,$arr 数组的元素并没有被修改为对 $val 的引用。
foreach 循环中正确的引用赋值方法
要在 foreach 循环中将数组元素绑定到另一个变量的引用,必须直接通过数组的键名来操作。
// 方法三:在 foreach 循环中通过键名直接赋值引用 (有效)
$arr = ['a' => 'AAA', 'b' => 'BBB']; // 重置数组以便测试
foreach ($arr as $ky => $dummy) { // $dummy 可以是按值传递,因为我们不直接操作它
$arr[$ky] = &$val; // 直接将 $arr[$ky] 赋值为 $val 的引用
}
echo "方法三(foreach中正确方法)后数组: " . print_r($arr, true) . "
"; // Array ( [a] => OOOOOO [b] => OOOOOO )在这个正确的示例中,我们不再尝试重新绑定 $vl,而是直接通过 $arr[$ky] 访问原始数组元素,并将其赋值为 $val 的引用。这样就能够达到预期的效果。
总结与注意事项
- 理解引用的本质:引用是别名,而不是独立的变量副本。$a = &$b; 意味着 $a 和 $b 指向同一块数据。
- foreach 引用别名行为:当 foreach ($arr as &$vl) 时,$vl 成为当前数组元素的别名。对 $vl 的直接值修改 ($vl = 'new_value';) 会影响原数组元素。
- 重新绑定引用的陷阱:在 foreach 循环内部,$vl = &$new_var; 这样的操作会改变 $vl 这个别名变量所指向的目标,但不会影响到它之前所引用的原始数组元素。原始数组元素将保持不变。
- 正确操作:如果需要在 foreach 循环中将数组元素绑定到另一个变量的引用,务必通过 $array[$key] = &$new_var; 的形式直接操作原始数组元素。
通过深入理解PHP中引用的工作机制,特别是在 foreach 循环中的行为差异,可以有效避免常见的编程陷阱,编写出更健壮、更符合预期的代码。











