资讯专栏INFORMATION COLUMN

PHP 循环中「引用」引发的奇怪问题

Riddler / 309人阅读

摘要:本文整理自网站上的一篇文章在循环中,如果使用引用会引发非常奇怪的行为这是的一个吗问题在我写一个简单的脚本时,发生了一些非常奇怪的现象。在中,如果一个内存空间是被引用的,那么当改变它的时候是直接改变这块内存空间的值。

本文整理自 stackoverflow 网站上的一篇文章 Strange behaviour after loop by reference - Is this a PHP bug? —— 在 PHP 循环中,如果使用 引用 会引发非常奇怪的行为 - 这是 PHP 的一个 bug 吗?

问题

在我写一个简单的 PHP 脚本时,发生了一些非常奇怪的现象。下面是我的代码,为了清楚的表达我的意思,我特意去掉了一些不必要的代码:



输出如下:

Array
(
    [0] => foo
    [1] => bar
    [2] => baz
)
Array
(
    [0] => foo
    [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

当第一个循环运行时,我们可以看到这样的输出:

foo
Array ( [0] => foo [1] => bar [2] => baz )

bar
Array ( [0] => foo [1] => bar [2] => baz )

baz
Array ( [0] => foo [1] => bar [2] => baz )

在循环结束后,$item$arr[2] 指向同一个内存区域。

当第二个循环运行时,我们看到这样的输出:

foo
Array ( [0] => foo [1] => bar [2] => foo )

bar
Array ( [0] => foo [1] => bar [2] => bar )

bar
Array ( [0] => foo [1] => bar [2] => bar )

在这次循环中,需要注意随着每次 $item 被赋予一个新值, $arr[2] 也会被赋值为和 $item 相同的值,因为它们都仍然指向相同的内存空间(译注:原文写的是 $arr[3],疑为原作者笔误。@justjavac)。 当循环到达数组的第三个值时,它包含的值是 bar,因为它的值在前两次循环中,被修改了。

还有疑问

也许你觉得,我仅仅是执行了一个空循环 foreach ($arr as &$item){},循环体里面什么都没有做,为什么数组元素却改变了?

可能你觉得这个代码应该等价于

for ($i = 0; $i < count($arr); $i++) { 
    // do nothing
}

其实不对,代码应该等价于:

for ($i = 0; $i < count($arr); $i++) { 
    $item = $arr[$i]; 
}

也就是说, 在 foreach 循环中,隐含了一个赋值运算,唯一不同的时, 在赋值过程中,我们使用了引用,所以在第一个循环中,无意中修改了正在循环的数组内部的元素。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/20609.html

相关文章

  • PHP 「自增、自减」运算引发奇怪问题

    摘要:在的官方手册中写道支持风格的前后递增与递减运算符。第一个注意事递增递减运算符不影响布尔值。递增递减布尔值递增递减在处理字符变量的算数运算时,沿袭了的习惯,而非的。还有一个注意事项递增递减其他字符变量则无效,原字符串没有变化。 在 PHP 的官方手册中写道: PHP 支持 C 风格的前/后递增与递减运算符。 第一个注意事:递增/递减运算符不影响布尔值。递减 NULL 值也没有...

    madthumb 评论0 收藏0
  • foreach遍历过程奇怪现象(PHP5)

    摘要:中基础中的三大坑,遍历,引用机制,数组。今天我们在讲讲中的一些奇怪现象。本文适合有一定基础的。运行流程共用一个结构体开始遍历数组,进行判断,拷贝数组是一个新的结构体,操作的是新的结构体。那么遍历数组时,全程与原数组无关。 PHP中基础中的三大坑,foreach遍历,引用机制&,数组。 今天我们在讲讲foreach中的一些奇怪现象。 在讲解之前,可以先看看我其他相关的文章,属于同一个大的...

    kgbook 评论0 收藏0
  • php底层原理之垃圾回收机制

    摘要:总结垃圾回收机制以的引用计数机制为基础以前只有该机制同时使用根缓冲区机制,当发现有存在循环引用的时,就会把其投入到根缓冲区,当根缓冲区达到配置文件中的指定数量后,就会进行垃圾回收,以此解决循环引用导致的内存泄漏问题开始引入该机制 php垃圾回收机制,对于PHPer来说是一个不陌生但是又不是很熟悉的内容。那么php是怎么实现对不需要的内存进行回收的呢? php变量的内部存储结构 首先还是...

    light 评论0 收藏0
  • 增量部署class文件引发血案

    摘要:背景项目中通过远程调用服务框架调用了许多其它的服务其中有一个服务需要升级其升级不是版本上的升级而是整个服务重新取了一个名字使用的也是全新的包但是调用的方法没有改变因此在升级时只是在调用服务类中修改了调用地址和调用返回实体由改为该中返回该调用 背景 项目中通过远程调用服务框架调用了许多其它的服务,其中有一个服务wx/subscribe/contract/CircleService 需要升...

    lolomaco 评论0 收藏0
  • 十个 PHP 开发者最容易犯错误

    摘要:这种行为比最初出现的问题更为棘手,同时也是一种常见的错误源。这意味着这个数组的一份拷贝将会被返回,因此被调函数与调用者所访问的数组并不是同样的数组实例。 showImg(https://segmentfault.com/img/bV7reP?w=620&h=620); PHP 语言让 WEB 端程序设计变得简单,这也是它能流行起来的原因。但也是因为它的简单,PHP 也慢慢发展成一个相对...

    fireflow 评论0 收藏0

发表评论

0条评论

Riddler

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<