我经常遇到这样的情况:处理可以是数组或空变量的数据,并将这些数据提供给一些foreach。

$values = get_values();

foreach ($values as $value){
  ...
}

当你给foreach提供不是数组的数据时,你会得到一个警告:

警告:在[…]中为foreach()提供的参数无效

假设不可能重构get_values()函数以总是返回一个数组(向后兼容,没有可用的源代码,无论其他原因),我想知道哪种是避免这些警告的最干净和最有效的方法:

将$值转换为数组 初始化数组的$values 用if包foreach 其他(请建议)


当前回答

foreach ($arr ?: [] as $elem) {
    // Do something
}

这并不检查它是否是一个数组,但如果变量为null或空数组则跳过循环。

从PHP 7.0更新,你应该使用空合并操作符:

foreach ($arr ?? [] as $elem) {
    // Do something
}

这将解决评论中提到的警告(这里有一个方便的表,比较?:和??输出)。

其他回答

<?php
 if (isset($_POST['checkbox'])) {
    $checkbox = $_POST['checkbox'];
    foreach ($checkbox as $key) {
       echo $key . '<br>';
    }
}

?>
    <input type="checkbox" name="checkbox[]" value="red">red <br>
    <input type="checkbox" name="checkbox[]" value="green">green <br>

这个怎么样?干净多了,都是单线的。

foreach ((array) $items as $item) {
 // ...
 }

请不要依赖铸造作为解决方案, 尽管其他人认为这是防止错误的有效选择,但它可能会导致另一个错误。

注意:如果您希望返回特定形式的数组,这可能会使您失败。为此需要进行更多的检查。

例如,将一个布尔值强制转换为数组(array)bool,将不会得到一个空数组,而是得到一个包含布尔值为int类型的元素的数组:[0=>0]或[0=>1]。

我编写了一个快速测试来呈现这个问题。 (这里有一个备份测试,以防第一个测试url失败。)

包括测试:null, false, true,类,数组和未定义。


在foreach中使用输入之前一定要测试它。建议:

Quick type checking: $array = is_array($var) or is_object($var) ? $var : [] ; Type hinting arrays in methods before using a foreach and specifying return types Wrapping foreach within if Using try{}catch(){} blocks Designing proper code / testing before production releases To test an array against proper form you could use array_key_exists on a specific key, or test the depth of an array (when it is one !). Always extract your helper methods into the global namespace in a way to reduce duplicate code

这个解决方案怎么样:

$type = gettype($your_iteratable);
$types = array(
    'array',
    'object'
);

if (in_array($type, $types)) {
    // foreach code comes here
}

如果get_value()为空,那么定义一个空数组作为回退呢? 我想不出一条最短的路。

$values = get_values() ?: [];

foreach ($values as $value){
  ...
}