检查数组是否为多维?

时间:2020-03-06 14:50:27  来源:igfitidea点击:
  • 检查数组是原始值的平面数组还是多维数组的最有效方法是什么?
  • 有什么方法可以做到这一点,而无需实际遍历数组并在其每个元素上运行is_array()

解决方案

我们可以在第一个元素上检查is_array(),假设如果数组的第一个元素是数组,那么其余元素也是。

此函数将返回int数组维数(从此处窃取)。

function countdim($array)
{
   if (is_array(reset($array))) 
     $return = countdim(reset($array)) + 1;
   else
     $return = 1;

   return $return;
}

简短的答案是,如果"第二维"可能在任何地方,至少没有隐式循环就无法做到。如果必须在第一项中,则只需执行

is_array($arr[0]);

但是,我能找到的最有效的通用方法是在数组上使用一个foreach循环,每当找到一个命中点时就会短路(至少隐式循环比直接的for()更好):

$ more multi.php
<?php

$a = array(1 => 'a',2 => 'b',3 => array(1,2,3));
$b = array(1 => 'a',2 => 'b');
$c = array(1 => 'a',2 => 'b','foo' => array(1,array(2)));

function is_multi($a) {
    $rv = array_filter($a,'is_array');
    if(count($rv)>0) return true;
    return false;
}

function is_multi2($a) {
    foreach ($a as $v) {
        if (is_array($v)) return true;
    }
    return false;
}

function is_multi3($a) {
    $c = count($a);
    for ($i=0;$i<$c;$i++) {
        if (is_array($a[$i])) return true;
    }
    return false;
}
$iters = 500000;
$time = microtime(true);
for ($i = 0; $i < $iters; $i++) {
    is_multi($a);
    is_multi($b);
    is_multi($c);
}
$end = microtime(true);
echo "is_multi  took ".($end-$time)." seconds in $iters times\n";

$time = microtime(true);
for ($i = 0; $i < $iters; $i++) {
    is_multi2($a);
    is_multi2($b);
    is_multi2($c);
}
$end = microtime(true);
echo "is_multi2 took ".($end-$time)." seconds in $iters times\n";
$time = microtime(true);
for ($i = 0; $i < $iters; $i++) {
    is_multi3($a);
    is_multi3($b);
    is_multi3($c);
}
$end = microtime(true);
echo "is_multi3 took ".($end-$time)." seconds in $iters times\n";
?>

$ php multi.php
is_multi  took 7.53565130424 seconds in 500000 times
is_multi2 took 4.56964588165 seconds in 500000 times
is_multi3 took 9.01706600189 seconds in 500000 times

隐式循环,但是一旦找到匹配项,我们就无法短路...

$ more multi.php
<?php

$a = array(1 => 'a',2 => 'b',3 => array(1,2,3));
$b = array(1 => 'a',2 => 'b');

function is_multi($a) {
    $rv = array_filter($a,'is_array');
    if(count($rv)>0) return true;
    return false;
}

var_dump(is_multi($a));
var_dump(is_multi($b));
?>

$ php multi.php
bool(true)
bool(false)

对于PHP 4.2.0或者更高版本:

function is_multi($array) {
    return (count($array) != count($array, 1));
}

两次使用count();在默认模式下一次,在递归模式下一次。如果值匹配,则该数组不是多维数组,因为多维数组将具有更高的递归计数。

if (count($array) == count($array, COUNT_RECURSIVE)) 
{
  echo 'array is not multidimensional';
}
else
{
  echo 'array is multidimensional';
}

PHP 4.2.0中添加了该选项第二个值mode。从PHP文档:

If the optional mode parameter is set to COUNT_RECURSIVE (or 1), count() will recursively count the array. This is particularly useful for counting all the elements of a multidimensional array. count() does not detect infinite recursion.

但是,该方法无法检测到" array(array())"。

is_array($arr[key($arr)]);

没有循环,简单明了。

不仅可以使用不能包含0的数值数组,还可以与关联数组一起使用(就像上一个示例中的警告一样,如果数组没有0的话)。

function isMultiArray(array $value)
{
    return is_array(reset($value));
}