使用 foreach 使用引用变量需要注意的问题

先看代码

$arr = [1, 2, 3];
foreach($arr as &$value){

}

foreach($arr as $value){

}

print_r($arr);

// 预期
Array
(
    [0] => 1
    [1] => 2
    [2] => 3
)

// 实际
Array
(
    [0] => 1
    [1] => 2
    [2] => 2
)

foreach 在执行的时候,会依次把数组 $arr 键的值赋值给 $value,并且 $value 并不是局部变量,它存在于全局中。


所以在第二次 foreach 执行的时候,此时的 $value = &$arr[2],由于它的内存地址指向了 $arr[2],所以在第二次的 foreach 循环中,每一次对 $value 的赋值,其实都是在改变 $arr[2] 的值。

$arr = [1, 2, 3];
foreach($arr as &$value){
    // 第一次循环  $value = &$arr[0];
    // 第二次循环  $value = &$arr[1];
    // 第三次循环  $value = &$arr[2];
}
// 循环结束后由于 $value 处于全局中,它依然指向 $arr[2] 
// 于是:
foreach($arr as $value){
    // 第一次循环 $value = $arr[0];
    // 实际上     $arr[2] = $arr[0];

    // 第二次循环  $value = $arr[1];
    // 实际上     $arr[2] = $arr[1];

    // 第三次循环  $value = $arr[2];
    // 实际上     $arr[2] = $arr[2];
    // 此时 $arr[2] 与 $arr[1] 相等
}

要解决这个问题也很简单

$arr = [1, 2, 3];
foreach($arr as &$value){

}
// 每次循环引用数组值的时候,循环之后要将 $value 释放掉
unset($value);
foreach($arr as $value){

}

print_r($arr);

// 输出
Array
(
    [0] => 1
    [1] => 2
    [2] => 3
) 

这应该是 PHP新手 经常忽略的问题,要记住 foreach 并不像 function 一样产生作用域,他的变量处在全局中。

记录一下。

本作品采用《CC 协议》,转载必须注明作者和本文链接
悲观者永远正确,乐观者永远前行。
《L03 构架 API 服务器》
你将学到如 RESTFul 设计风格、PostMan 的使用、OAuth 流程,JWT 概念及使用 和 API 开发相关的进阶知识。
《G01 Go 实战入门》
从零开始带你一步步开发一个 Go 博客项目,让你在最短的时间内学会使用 Go 进行编码。项目结构很大程度上参考了 Laravel。
讨论数量: 4
JaguarJack

老生常谈了

2年前 评论
sreio
$arr = [1, 2, 3];
foreach($arr as &$value){

}
foreach($arr as $value1){

}

print_r($arr);

// 输出
Array
(
    [0] => 1
    [1] => 2
    [2] => 3
) 
2年前 评论
sreio (作者) 2年前

感谢了,之前看各种文章还是云里雾里的。看到了文章里面第二个 foreach 的注释秒懂了。 :+1:

2年前 评论

讨论应以学习和精进为目的。请勿发布不友善或者负能量的内容,与人为善,比聪明更重要!