这有点类似于问题:如何确定 foreach 循环中的第一次和最后一次迭代?,但是,这个 +10 岁的问题非常有array
针对性,并且没有一个答案与可以循环许多不同类型的事实兼容。
给定一个可以在 PHP >= 7.4中迭代PDOStatement
的东西(数组、迭代器、生成器、DatePeriod
...、对象属性等)的循环,我们如何以有效的方式触发需要在循环,但仅在进入循环的情况下?
一个典型的用例可能是生成 HTML 列表:
<ul>
<li>...</li>
<li>...</li>
...
</ul>
<ul>
并且只有在存在某些元素时才</ul>
必须打印。
这些是我到目前为止发现的约束:
empty()
: 不能用于生成器/迭代器。each()
: 已弃用。iterator_to_array()
打败了发电机的优势。- 在循环内部和之后测试的布尔标志不被认为是有效的,因为它会导致在每次迭代时执行该测试,而不是在循环开始和结束时执行一次。
- 虽然在上面的示例中可以使用输出缓冲或字符串连接来生成输出,但它不适合循环不会产生任何输出的情况。(感谢@barmar的额外想法)
以下代码片段总结了我们可以迭代的许多不同类型foreach
,它可以用作提供答案的开始:
<?php
// Function that iterates on $iterable
function iterateOverIt($iterable) {
// How to generate the "<ul>"?
foreach ($iterable as $item) {
echo "<li>", $item instanceof DateTime ? $item->format("c") : (
isset($item["col"]) ? $item["col"] : $item
), "</li>\n";
}
// How to generate the "</ul>"?
}
// Empty array
iterateOverIt([]);
iterateOverIt([1, 2, 3]);
// Empty generator
iterateOverIt((function () : Generator {
return;
yield;
})());
iterateOverIt((function () : Generator {
yield 4;
yield 5;
yield 6;
})());
// Class with no public properties
iterateOverIt(new stdClass());
iterateOverIt(new class { public $a = 7, $b = 8, $c = 9;});
$db = mysqli_connect("localhost", "test", "test", "test");
// Empty resultset
iterateOverIt($db->query("SELECT 0 FROM DUAL WHERE false"));
iterateOverIt($db->query("SELECT 10 AS col UNION SELECT 11 UNION SELECT 12"));
// DatePeriod generating no dates
iterateOverIt(new DatePeriod(new DateTime("2020-01-01 00:00:00"), new DateInterval("P1D"), new DateTime("2020-01-01 00:00:00"), DatePeriod::EXCLUDE_START_DATE));
iterateOverIt(new DatePeriod(new DateTime("2020-01-01 00:00:00"), new DateInterval("P1D"), 3, DatePeriod::EXCLUDE_START_DATE));
这样的脚本应产生以下输出:
<ul>
<li>1</li>
<li>2</li>
<li>3</li>
</ul>
<ul>
<li>4</li>
<li>5</li>
<li>6</li>
</ul>
<ul>
<li>7</li>
<li>8</li>
<li>9</li>
</ul>
<ul>
<li>10</li>
<li>11</li>
<li>12</li>
</ul>
<ul>
<li>2020-01-02T00:00:00+00:00</li>
<li>2020-01-03T00:00:00+00:00</li>
<li>2020-01-04T00:00:00+00:00</li>
</ul>