PHP sizeof或count不正确

时间:2012-10-08 23:52:38

标签: php arrays multidimensional-array

以下是从XML文档转换的数组的print_r:

Array
    (
        [layer1] => Array
            (
                [item] => Array
                    (
                        [item-id] => 1886731
                        [item-name] => Bad Dog
                        [category] => pets
                        [link] = http://www.baddog.com/
                    )
            [total-matched] => 1
        )
    )

对于上面的数组,sizeof($ myarray [layer1] [item])应该返回1,但它返回4.如果有多个“item”项,我会得到正确数量的“item”项。无论我使用“sizeof”还是“count”,都会发生同样的错误。当只有一个项目时,如何让PHP返回“1”?

因此,如果有一个项目,我无法使用数组[layer1] [item] [0] [item-name]访问item-name,我必须使用array [layer1] [item] [item-名]。

1 个答案:

答案 0 :(得分:6)

世界之间存在差异:

$array1 = array(
  'layer1' => array(
    'item' => array(
      '0' => array(
        'item-id' => 123123,
        'item-name' => 'Bad Dog',
        'category' => 'pets',
        'link' => 'http://www.baddog.com/',
      ),
    )
  )
);

$array2 = array(
  'layer1' => array(
    'item' => array(
       'item-id' => 123123,
       'item-name' => 'Bad Dog',
       'category' => 'pets',
       'link' => 'http://www.baddog.com/',
    )
  )
);

基本上它们是不同的结构,PHP返回以下内容是正确的:

count($array1['layer1']['item']);
/// = 1 (count of items in the array at that point)

count($array2['layer1']['item']);
/// = 4 (count of items in the array at that point)

如果您希望获得对您的应用有意义的['layer1']['item']计数,您将始终需要['layer1']['item']成为包含多个数组结构的数组...即$array1以上。这就是为什么我要问什么是生成你的数组结构,因为 - 无论它是什么 - 它基本上是根据项目的数量智能地堆叠你的数组数据,这是你不想要的。

可能导致数组以这种方式堆叠的代码通常类似于以下内容:

/// $array = array(); /// this is implied (should be set elsewhere)

$key = 'test';
$newval = 'value';

/// if we are an array, add a new item to the array
if ( is_array($array[$key]) ) {
  $array[$key][] = $newval;
}
/// if we have a previous non-array value, convert to an array
/// containing the previous and new value
else if ( isset($array[$key]) ) {
  $array[$key] = array($array[$key],$newval);
}
/// if nothing is set, set the $newval
else {
  $array[$key] = $newval;
}

基本上如果你继续调用上面的代码,并在每次运行后跟踪,你会看到以下结构:

$array == 'value';

然后

$array == array(0 => 'value', 1 => 'value');

然后

$array == array(0 => 'value', 1 => 'value', 2 => 'value');

这是导致问题的第一步,$array = 'value';位。如果你稍微修改一下代码就可以解决这个问题:

/// $array = array(); /// this is implied (should be set elsewhere)
$key = 'test';
$newval = 'value';

/// if we are an array, add a new item to the array
if ( is_array($array[$key]) ) {
  $array[$key][] = $newval;
}
/// if nothing is set, set the $newval as part of an subarray
else {
  $array[$key] = array($newval);
}

正如你所看到的,我所做的就是删除itermediate if statement,并确保当我们发现没有设置初始值时,我们总是创建一个数组。以上将创建一个您可以随时计算的结构,并知道您推送到阵列的项目数。

$array == array(0 => 'value');

然后

$array == array(0 => 'value', 1 => 'value');

然后

$array == array(0 => 'value', 1 => 'value', 2 => 'value');

<强>更新

啊,我是这么认为的。所以数组是从XML生成的。在这种情况下,我收集到您正在使用预定义的库来执行此操作,因此修改代码是不可能的。正如其他人已经说过的那样,最好的办法是使用PHP可用的众多XML解析库之一:

http://www.uk.php.net/simplexml

http://www.uk.php.net/dom

使用这些系统时,您可以保留更多易于计算的对象结构。以上两者都支持xpath表示法,它可以让你计算项目,甚至不需要抓住任何数据。

更新2

在您给出的函数之外,这是导致数组以导致问题的方式堆叠的部分:

$children = array();
$first = true;
foreach($xml->children() as $elementName => $child){
    $value = simpleXMLToArray($child,$attributesKey, $childrenKey,$valueKey);
    if(isset($children[$elementName])){
        if(is_array($children[$elementName])){
            if($first){
                $temp = $children[$elementName];
                unset($children[$elementName]);
                $children[$elementName][] = $temp;
                $first=false;
            }
            $children[$elementName][] = $value;
        }else{
            $children[$elementName] = array($children[$elementName],$value);
        }
    }
    else{
        $children[$elementName] = $value;
    }
}

修改将是:

$children = array();
foreach($xml->children() as $elementName => $child){
    $value = simpleXMLToArray($child,$attributesKey, $childrenKey,$valueKey);
    if(isset($children[$elementName])){
        if(is_array($children[$elementName])){
            $children[$elementName][] = $value;
        }
    }
    else{
        $children[$elementName] = array($value);
    }
}

这应该会阻止您的数组堆叠...但是如果您的代码的任何其他部分依赖于以前的结构,则此更改可能会破坏该代码。