PHP foreach 的 7 处更改:我仍然可以删除正在迭代的 they 数组中的项目吗?
PHP 7 changes to foreach: Can I still delete items in they array on which I'm iterating?
PHP 7 Backward-Incompatible Changes Document 对 foreach 的描述如下:
When used in the default by-value mode, foreach
will now operate on a copy of the array being iterated rather than the array itself. This means that changes to the array made during iteration will not affect the values that are iterated.
我试图理解这意味着什么,我的主要问题是这段代码在 PHP 7 中是否会像在 PHP 5.6 中一样工作?
foreach($array as $elementKey => $element) {
if ($element == 'x') {
unset($array[$elementKey]);
}
}
我的两个问题是:
此代码仍然有效吗?
如果是这样,您能解释一下(例如)PHP 7 中的这个新变化意味着什么吗?
编辑
我一直在重新阅读文档声明。我在想这意味着,如果您更改数组中较低位置的项目的值,那么当您在迭代中到达这些项目时,这些更改将不会存在。示例:
$array = ['x', 'y', 'z'];
$new = [];
foreach($array as $element) {
if ($element == 'x') {
$array[2] = 'a';
}
$new[] = $element;
}
print_r($new);
然而,当我 run this example 时,PHP 版本 似乎没有显示任何差异(尽管我以前从未使用过此工具,所以我不确定它是如何工作的)。
我意识到如果我通过引用来做,我会得到一个新的 a
。否则我不会。但是两个版本好像都是这样。
我真正需要知道的是什么是不兼容(通过示例)?
编辑 2
@NikiC 的 answer link suggested 提供了我正在寻找的故事的其余部分:
In most cases this change is transparent and has no other effect than better performance. However there is one occasion where it results in different behavior, namely the case where the array was a reference beforehand:
$array = [1, 2, 3, 4, 5];
$ref = &$array;
foreach ($array as $val) {
var_dump($val);
$array[2] = 0;
}
/* Old output: 1, 2, 0, 4, 5 */
/* New output: 1, 2, 3, 4, 5 */
Previously by-value iteration of reference-arrays was special cases. In this case no duplication occurred, so all modifications of the array during iteration would be reflected by the loop. In PHP 7 this special case is gone: A by-value iteration of an array will always keep working on the original elements, disregarding any modifications during the loop.
这个答案解释了罕见的 "special case" 版本之间在 foreach
对数组副本进行操作方面的不同之处。
这意味着您现在必须明确地说您想要引用您正在迭代的数组。
但是,在您的示例代码中,您无论如何都在引用根数组,因此无论您是否通过引用传递,它都会起作用。
<?php
$array = ['x', 'y', 'z'];
foreach($array as $elementKey => $element) {
if ($element=='x') {
unset($array[$elementKey]);
}
}
var_dump($array); // lists 'y' and 'z'
一个更好的例子。在这种情况下,我们在没有引用的情况下更改 foreach
中的值。因此,本地更改将丢失:
<?php
$array = ['x', 'y', 'z'];
foreach($array as $element) {
if ($element=='x') {
$element = 'a';
}
}
var_dump($array); // 'x', 'y', 'z'
Vs by reference,这里我们声明$element
是对数组元素的引用:
<?php
$array = ['x', 'y', 'z'];
foreach($array as &$element) {
if ($element=='x') {
$element = 'a';
}
}
var_dump($array); // 'a', 'y', 'z'
Here's a demo of this code running 在 PHP 的各种版本上。
PHP 7 Backward-Incompatible Changes Document 对 foreach 的描述如下:
When used in the default by-value mode,
foreach
will now operate on a copy of the array being iterated rather than the array itself. This means that changes to the array made during iteration will not affect the values that are iterated.
我试图理解这意味着什么,我的主要问题是这段代码在 PHP 7 中是否会像在 PHP 5.6 中一样工作?
foreach($array as $elementKey => $element) {
if ($element == 'x') {
unset($array[$elementKey]);
}
}
我的两个问题是:
此代码仍然有效吗?
如果是这样,您能解释一下(例如)PHP 7 中的这个新变化意味着什么吗?
编辑
我一直在重新阅读文档声明。我在想这意味着,如果您更改数组中较低位置的项目的值,那么当您在迭代中到达这些项目时,这些更改将不会存在。示例:
$array = ['x', 'y', 'z'];
$new = [];
foreach($array as $element) {
if ($element == 'x') {
$array[2] = 'a';
}
$new[] = $element;
}
print_r($new);
然而,当我 run this example 时,PHP 版本 似乎没有显示任何差异(尽管我以前从未使用过此工具,所以我不确定它是如何工作的)。
我意识到如果我通过引用来做,我会得到一个新的 a
。否则我不会。但是两个版本好像都是这样。
我真正需要知道的是什么是不兼容(通过示例)?
编辑 2
@NikiC 的 answer link suggested 提供了我正在寻找的故事的其余部分:
In most cases this change is transparent and has no other effect than better performance. However there is one occasion where it results in different behavior, namely the case where the array was a reference beforehand:
$array = [1, 2, 3, 4, 5];
$ref = &$array;
foreach ($array as $val) {
var_dump($val);
$array[2] = 0;
}
/* Old output: 1, 2, 0, 4, 5 */
/* New output: 1, 2, 3, 4, 5 */
Previously by-value iteration of reference-arrays was special cases. In this case no duplication occurred, so all modifications of the array during iteration would be reflected by the loop. In PHP 7 this special case is gone: A by-value iteration of an array will always keep working on the original elements, disregarding any modifications during the loop.
这个答案解释了罕见的 "special case" 版本之间在 foreach
对数组副本进行操作方面的不同之处。
这意味着您现在必须明确地说您想要引用您正在迭代的数组。
但是,在您的示例代码中,您无论如何都在引用根数组,因此无论您是否通过引用传递,它都会起作用。
<?php
$array = ['x', 'y', 'z'];
foreach($array as $elementKey => $element) {
if ($element=='x') {
unset($array[$elementKey]);
}
}
var_dump($array); // lists 'y' and 'z'
一个更好的例子。在这种情况下,我们在没有引用的情况下更改 foreach
中的值。因此,本地更改将丢失:
<?php
$array = ['x', 'y', 'z'];
foreach($array as $element) {
if ($element=='x') {
$element = 'a';
}
}
var_dump($array); // 'x', 'y', 'z'
Vs by reference,这里我们声明$element
是对数组元素的引用:
<?php
$array = ['x', 'y', 'z'];
foreach($array as &$element) {
if ($element=='x') {
$element = 'a';
}
}
var_dump($array); // 'a', 'y', 'z'
Here's a demo of this code running 在 PHP 的各种版本上。