Heim >Backend-Entwicklung >PHP-Tutorial >php foreach 中使用引用不当,撑爆内存

php foreach 中使用引用不当,撑爆内存

WBOY
WBOYOriginal
2016-06-06 20:48:551527Durchsuche

今天一个同事,写了一段类似下面的代码:

<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的内部实现机制造成的,求解释!!

Stellungnahme:
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn