PHP行为和数组指针

4

我正在阅读 PHP 手册(具体来说是 each() 函数),并发现了以下警告:

注意
因为将数组分配给另一个变量会重置原始数组的指针,所以如果我们在循环内部将 $fruit 赋值给另一个变量,上面的示例将导致无限循环。

以下是一个示例:

<?php
$fruit = array('a' => 'apple', 'b' => 'banana', 'c' => 'cranberry');

reset($fruit);
while (list($key, $val) = each($fruit)) {
    echo "$key => $val\n";
}
?>

好的,这很有道理。但我决定做一个简单的测试:

<?php
    $fruit = array('a' => 'apple', 'b' => 'banana', 'c' => 'cranberry');

    foreach ($fruit as $key => $name) {
        printf("[%s] => [%s]\n", $key, $name);
    }

    $fruit2 = $fruit;
    echo current($fruit);
?>

预期结果是:指针已被重置。我的问题是,指针是否仅在数组结束后重置?
例如:
<?php
    $fruit = array('a' => 'apple', 'b' => 'banana', 'c' => 'cranberry');

    foreach ($fruit as $key => $name) {
        printf("[%s] => [%s]\n", $key, $name);
    }

    reset($fruit);
    next($fruit)."\n";
    $fruit2 = $fruit;
    echo current($fruit);
?>

指针仍然停留在第二个数组元素 ('b' => 'banana')。 这种行为是这种语言的特点吗?谢谢并对我的糟糕英语表示抱歉。

在 PHP 5 中,foreach() 函数的行为似乎出现了问题。但在 PHP 7 中已经修复:https://github.com/tpunt/PHP7-Reference#fixes-to-foreachs-behaviour - axiac
2个回答

5
这种行为特性属于语言本身吗?
在 PHP 数组中,“指针”的含义与“指针”(在 C/C++ 或其他直接访问内存的语言中)的一般含义不同。
PHP 中没有指针。数组 数据类型在其包含的值列表中保留了一个光标。它被称为数组的内部指针,并通过 reset()next()prev()end()each() 函数进行修改,也可能有其他函数。可以使用它来迭代数组,如下所示:
$array = array(1, 2, 3);
while (list($key, $val) = each($array)) {
    echo($key.' => '.$val."\n");
}

没有一种可靠的方法可以使用 next()prev() 迭代数组,因为当没有更多元素可迭代时,它们返回 FALSE,但是当将值 FALSE 存储为数组元素时,它们也返回 FALSE
如果您只需要从数组开头(或结尾)分析几个项目,则它们可能很有用。例如,假设我们有一个由函数返回的整数数组,并且我们需要获取第一个非零值。
但是,甚至可以使用 foreach() 更轻松地实现此目标:
$array = array(0, 0, 0, 2, 0, 1, 0, 3);
foreach ($array as $val) {
    if ($val != 0) {
        break;
    }
}
echo($val);           // prints "2"

array_shift()

$array = array(0, 0, 0, 2, 0, 1, 0, 3);
do {
    $val = array_shift($array);
    if ($val != 0) {
        break;
    }
} while(count($array));
echo($val);           // prints "2"

预期结果:指针已重置。我的问题是数组结束后指针是否才被重置? foreach()的文档有误。也许在PHP 3和PHP 4上是正确的,但我认为自从PHP 5引入迭代器以来,foreach()的行为已经改变(更好了)。
文档中写道:
当foreach开始执行时,内部数组指针会自动重置为数组的第一个元素。这意味着您不需要在foreach循环之前调用reset()。
由于foreach依赖于内部数组指针,因此在循环中更改它可能会导致意外的行为。
一个简单的测试与该声明相矛盾:
$array = array(1, 3, 5, 7, 9);

foreach ($array as $val1) {
    foreach ($array as $val2) {
        echo('$val1='.$val1.'; $val2='.$val2.'; ');
    }
    echo("\n");
}

它可以正常工作。如果foreach()使用内部数组指针,则不应该工作。它可能会创建一个指针的副本。

您也可以尝试在foreach()内部使用current()next()prev()reset(),您会得到令人惊讶且有时不一致的结果。

最好使用foreach()对数组进行迭代,并且不要以任何方式依赖内部指针。

然而,当您需要获取数组的第一个和最后一个元素而不必担心键时,函数reset()end()非常方便。


谢谢您的解释!在这方面,PHP有点不太清晰。有时手册也无法提供帮助。 - Luis

2

是的,指针只有在数组结束后才会重置。由于foreach将指针指向数组的末尾,因此在$fruit2 = $fruit;之后它会自动重置,并且如果您使用下面的代码手动到达数组末尾,则它也会重置:

<?php
$fruit = array('a' => 'apple', 'b' => 'banana', 'c' => 'cranberry');

foreach ($fruit as $key => $name) {
    printf("[%s] => [%s]\n", $key, $name);
}

reset($fruit);
next($fruit)."\n";
next($fruit)."\n";
next($fruit)."\n";
$fruit2 = $fruit;
echo current($fruit);
?>

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接