php中使用foreach进行引用赋值时的问题


 最近在阅读公司的源码的时候,发现公司好几处地方使用foreach的引用(&)之后都会再写一行代码进行处理,一开始不明白,觉得有点多余,后来在某些时候使用的时候发现出现了一些问题,下面举例说明下。

  代码如下:

<?php
$ar = array(1, 2, 3);
var_dump($ar);

foreach ($ar as &$v) {}
var_dump($ar);

foreach ($ar as $v) {}
var_dump($ar);

  上面的代码咋看起来挺简单的,大家可以猜猜最后的输出结果是什么。

  上面的结果分别如下:

  

_W94_[{F{CE%FJXD4TYWATV.png

图片.png

XO[J`S4(P}O}2]ZW`HAC0@A.png

  我们会发现,最后一次打印出来的数组的第三个值变了,为什么会这样子呢?

  在执行使用引用的 foreach 时, 一开始, $v 指向 $ar[0] 的存储空间,空间内存储着 1 , foreach 结束时, $v 指向 $ar[2] 的存储空间,空间内存储着 3 。 下面要开始执行第二个 foreach 了,注意和第一个 foreach 不同, 第二个 foreach 没有使用引用,那么就是赋值方式, 即将 $ar 的值依次 赋值 给 $v 。 进行到第一个元素时,要将 $ar[0] 赋值给 $v 。 问题就在这里,由于刚刚执行完第一个 foreach, $v 不是一个新变量,而是已经存在的、指向 $ar[2] 的那个 引用 , 如此一来,对 $v 进行赋值的时候,就将 $ar[0] = 1 写入了 $ar[2] 的实际存储空间, 相当于对 $ar[2] 进行赋值。 依此类推,第二个 foreach 执行的结果, 就是数组的最后一个元素变成了倒数第二个元素的值。
  如果说这是一个错误,那么错误的原因就在于对引用变量的使用。 当引用变量指向和其他变量时,改变引用变量的值当然会影响到他指向的其他变量。 单独说谁都明白,但在这个 foreach 例子中,凑巧了, 同一个变量两次被使用,前一次是引用的身份,后一次是普通变量身份, 就产生了意料之外的效果。 PHP 的开发者也认为,这种情况属于语言特性造成的,不是 bug。 的确,如果要修复这个问题,一种方法是对 foreach 进行特殊处理之外, 另外一种就是限制 foreach 中 $v 的作用域, 这两种方式都与目前 PHP 的语言特性不符,开发人员不愿改, 但还是在 官方文档 中用 Warning 进行了说明。


  解决方法:

  下面我来说说官方推荐的一种方法。就是在使用了引用的 foreach 之后, unset 掉 $v。上面的例子如下:

<?php
$ar = array(1, 2, 3);
var_dump($ar);
foreach ($ar as &$v) {}
unset($v);
foreach ($ar as $v) {}
var_dump($ar);

其输出的结果如下:

  05KXDYCWB61GHY~37L6]{8N.png

K`O3HCTQNRUH@SM}KPDD)PU.png

A3_Y]Y$AQ]DU4)HGZE71T9Q.png




头像
QQ登录: