今天一个同事,写了一段类似下面的代码:
<code class="lang-php">$a = array(1, 2, 3); foreach($a as $k => &$v) { $a[] = $v; } </code>
默认的 128M 内存直接被撑爆了, 原因很简单:每循环一次,都将 $a
数组增加一个 k-v,导致陷入一个无限循环中, 内存自然不够用了。
关键不是这里, 关键是将 $k => &$v
改为 $k => $v
后, 这个循环又可以正常执行了。
查阅 PHP 手册中, 有一句:
"Note:除非数组是被引用,foreach 所操作的是指定数组的一个拷贝,而不是该数组本身。"
是否因为第二种写法 的 foreach
操作的是数组 $a
的拷贝,而第一种写法操作的是 $a
的引用或者其本身?
否则这个逻辑错误应该是两种写法都会体现出来才对。
今天一个同事,写了一段类似下面的代码:
<code class="lang-php">$a = array(1, 2, 3); foreach($a as $k => &$v) { $a[] = $v; } </code>
默认的 128M 内存直接被撑爆了, 原因很简单:每循环一次,都将 $a
数组增加一个 k-v,导致陷入一个无限循环中, 内存自然不够用了。
关键不是这里, 关键是将 $k => &$v
改为 $k => $v
后, 这个循环又可以正常执行了。
查阅 PHP 手册中, 有一句:
"Note:除非数组是被引用,foreach 所操作的是指定数组的一个拷贝,而不是该数组本身。"
是否因为第二种写法 的 foreach
操作的是数组 $a
的拷贝,而第一种写法操作的是 $a
的引用或者其本身?
否则这个逻辑错误应该是两种写法都会体现出来才对。
首先,肯定你的回答,原因也跟你说的一样,php里引用传递和值传递的区别。至于调试,我建议使用Xdebug调试一下即可,打印数组的zval结构体,代码如下:
<code class="lang-php"><?php $arr = array(1, 2, 3); foreach ($arr as $key => &$value) { $arr[] = $value; xdebug_debug_zval('arr'); echo "\n"; } </code>
可以看到zval的结构体的成员:arr: (refcount=3, isref=1),isref字段为1说明是arr的引用
我本该安心的复习算法,准备各种校招笔试的,但是无奈测试一下一个元素的数组,代码:
<code class="lang-php"><?php $arr = array(1); foreach ($arr as $key => &$value) { $arr[] = $value; xdebug_debug_zval('arr'); echo "\n"; } </code>
这个并没有像我想得一样进入死循环,考虑是否是foreach的内部实现机制造成的,求解释!!