PHP的foreach中使用引用时需要注意的一个问题和解决方法


Posted in PHP onMay 29, 2014

一、问题
先看一个例子:

<?php
$ar = array(1, 2, 3);
var_dump($ar);
foreach ($ar as &$v) {}
foreach ($ar as $v) {}
var_dump($ar);
?>
输出为:

array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
}
array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  &int(2)
}
???为什么没有进行赋值操作,数组最后一个元素的值却发生了改变呢?

我早就发现了这个问题,一开始以为是 PHP 的 bug,就扔着没管它, foreach 中不使用引用就没事, 用 foreach $k => $v 然后 $ar[$k] 来改变原始数组, 略微损失点效率。

二、分析

今天花了点时间,看了 参考 中的文章, 算是稍微明白一点了,原来是这个样子的:

在执行第一个使用引用的 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 执行的结果, 就是数组的最后一个元素变成了倒数第二个元素的值。 参考文章 2 中有详细的示意图。

如果说这是一个错误,那么错误的原因就在于对引用变量的使用。 当引用变量指向和其他变量时,改变引用变量的值当然会影响到他指向的其他变量。 单独说谁都明白,但在这个 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);
?>
运行结果:

array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
}
array(3) {
  [0]=>
  int(1)
  [1]=>
  int(2)
  [2]=>
  int(3)
}

参考

Bug #29992 foreach by reference corrupts the array:https://bugs.php.net/bug.php?id=29992
References and foreach:http://schlueters.de/blog/archives/141-References-and-foreach.html

PHP 相关文章推荐
php2html php生成静态页函数
Dec 08 PHP
php 正则 过滤html 的超链接
Jun 02 PHP
PHPThumb PHP 图片缩略图库
Mar 11 PHP
PHP ? EasyUI DataGrid 资料存的方式介绍
Nov 07 PHP
phpmyadmin显示utf8_general_ci中文乱码的问题终级篇
Apr 08 PHP
探讨:parse url解析URL,返回其组成部分
Jun 14 PHP
使用Linux五年积累的一些经验技巧
Jun 20 PHP
php 注册时输入信息验证器的实现详解
Jul 05 PHP
php使用curl访问https示例分享
Jan 17 PHP
PHP针对中英文混合字符串长度判断及截取方法示例
Mar 31 PHP
php实现页面纯静态的实例代码
Jun 21 PHP
浅谈使用 Yii2 AssetBundle 中 $publishOptions 的正确姿势
Nov 08 PHP
神盾加密解密教程(一)PHP变量可用字符
May 28 #PHP
CI框架开发新浪微博登录接口源码完整版
May 28 #PHP
PHP+javascript制作带提示的验证码源码分享
May 28 #PHP
微信支付开发教程(一)微信支付URL配置
May 28 #PHP
php中$美元符号与Zen Coding冲突问题解决方法分享
May 28 #PHP
php轻松实现中英文混排字符串截取
May 28 #PHP
分享一段php获取linux服务器状态的代码
May 27 #PHP
You might like
织梦模板标记简介
2007/03/11 PHP
php实现简单洗牌算法
2013/06/18 PHP
Yii2.0多文件上传实例说明
2017/07/24 PHP
PHP递归实现文件夹的复制、删除、查看大小操作示例
2017/08/11 PHP
点击显示指定元素隐藏其他同辈元素的方法
2014/02/19 Javascript
Jquery实现的角色左右选择特效
2014/05/21 Javascript
node.js中的fs.fstatSync方法使用说明
2014/12/15 Javascript
全面解析Bootstrap排版使用方法(标题)
2015/11/30 Javascript
Bootstarp风格的toggle效果分享
2016/02/23 Javascript
javascript经典特效分享 手风琴、轮播图、图片滑动
2016/09/14 Javascript
使用JS代码实现点击按钮下载文件
2016/11/12 Javascript
Vue.js第四天学习笔记
2016/12/02 Javascript
JQuery 选择器、DOM节点操作练习实例
2017/09/28 jQuery
swiper自定义分页器使用方法详解
2020/09/14 Javascript
微信小程序时间标签和时间范围的联动效果
2019/02/15 Javascript
vue响应式系统之observe、watcher、dep的源码解析
2019/04/09 Javascript
浅谈VUE防抖与节流的最佳解决方案(函数式组件)
2019/05/22 Javascript
如何给element添加一个抽屉组件的方法步骤
2019/07/14 Javascript
vue实现浏览器全屏展示功能
2019/11/27 Javascript
js实现随机抽奖
2020/03/19 Javascript
vue使用openlayers实现移动点动画
2020/09/24 Javascript
Python实现树莓派WiFi断线自动重连的实例代码
2017/03/16 Python
python网络编程调用recv函数完整接收数据的三种方法
2017/03/31 Python
5款Python程序员高频使用开发工具推荐
2019/04/10 Python
python+selenium select下拉选择框定位处理方法
2019/08/24 Python
python 通过邮件控制实现远程控制电脑操作
2020/03/16 Python
Canvas 像素处理之改变透明度的实现代码
2019/01/08 HTML / CSS
高品质和独特的产品世界:Creations and Collections
2018/01/07 全球购物
联想新西兰官方网站:Lenovo New Zealand
2018/10/30 全球购物
英语国培研修感言
2014/02/13 职场文书
大学信息公开实施方案
2014/03/09 职场文书
保护环境建议书
2014/03/12 职场文书
学生偷窃检讨书
2014/09/25 职场文书
公安机关查摆剖析材料
2014/10/10 职场文书
2016年少先队活动总结
2016/04/06 职场文书
CDPR谈《巫师》新作用虚幻5原因 称不会为Epic独占
2022/04/06 其他游戏