记录一次因为对PHP作用域理解不够导致的小坑。
自测需求的时候发现有一块地方数据很奇怪,要么该写的没有写入、要么数据被写入双份。剥离业务后的代码大概如下:
<?php $arr = [ ['is_checked'=>false,'k'=>1], ['is_checked'=>false,'k'=>2], ]; foreach ($arr as &$item) { if ($item['k']==1) { $item['is_checked'] = true; } } echo '<pre>'; foreach ($arr as $item) { if ($item['is_checked']) { print_r($item); } }
我预想中 上面的代码应该是只打印arr里的第一条记录,也就是['is_checked'=>true,'k'=1]
,然而实际运行发现打印的是这样的:
Array ( [is_checked] => 1 [k] => 1 ) Array ( [is_checked] => 1 [k] => 1 )
居然打印了两条记录,而且两条的k都是1。
断点调试的时候也发现,运行到第二个foreach里的时候 arr确实变成了这样的数组:
[ ['is_checked'=>true,'k'=>1], ['is_checked'=>true,'k'=>1], ]
仔细看代码,前面foreach的时候,循环里的变量是用的item,而且是取引用,后面的foreach也是item。我之前是认为这俩item的作用域是不重合的,也就是认为第一个foreach的作用域只在foreach代码块里(这点可能是受了golang变量作用域的影响)
然而从结果来看,两个item应该是一样的,也就是第二个循环里的item还是前一个循环里的item,而前一个循环里的item是对数组里元素取的引用,也就是说,第一个循环结束后,item还是指向$arr
的第二个元素。第二个foreach开始的时候,$arr
的第一个元素的值被赋给item,这样$arr
的第二个元素就被第一个元素覆盖了,所以产生了上面的结果。
来一段代码验证下:
<?php $arr1 = [1,2,3,4]; foreach ($arr1 as &$item) { //do nothing } $arr2 = ['a','b','c','d']; echo '<pre>'; foreach ($arr2 as $item) { print_r($arr1); }
输出结果:
Array ( [0] => 1 [1] => 2 [2] => 3 [3] => a ) Array ( [0] => 1 [1] => 2 [2] => 3 [3] => b ) Array ( [0] => 1 [1] => 2 [2] => 3 [3] => c ) Array ( [0] => 1 [1] => 2 [2] => 3 [3] => d )
这里应该算是比较基础的点吧。但是因为对作用域范围不够敏感,踩了个坑还排查了半天(实际业务代码较多,开始没想到是这里的问题)。
说下这里要注意的点吧
foreach
时候的循环变量尽量不要用同一个变量,尤其是涉及到取引用的- 循环变量取引用的,退出循环后,最好是unset掉,防止后面不小心改掉了该数据