8

「$needle」で指定されたすべてのセットの結果セットを返す再帰的な配列反復子関数を作成しようとしています。$needle = キー

これが私の機能です:

function recursive($needle, $array, $holder = array()) {
    foreach ($array as $key => $value) {
        if (gettype($value) == 'array') {
            if ($key != $needle) {
                recursive($needle, $value);
            } elseif ($key == $needle) {
                if (!empty($value)) {
                    array_push($holder, $value);
                }
            }
        }
    }
    return $holder;
}

!empty($value)しかし、入力配列には空のセットがありませんが、を指定しないと、すべての結果が返されるわけではなく、代わりにいくつかの空の結果が返されます。私は何を間違っていますか?

4

3 に答える 3

17

PHP には標準のRecursive Iterator APIがあるため、車輪を再発明する必要はありません。

//$array is your multi-dimensional array
$result   = [];
$search   = 'foo';
$iterator = new RecursiveIteratorIterator(
    new RecursiveArrayIterator(
        $array,
        RecursiveArrayIterator::CHILD_ARRAYS_ONLY
    )
); 

foreach($iterator as $key=>$value)
{
   if($search==$key && $value!=='')
   {
      $result[] = $value;
   }
}

-キーで値を検索しているため、通常$valueはサブセクション全体が保持されることに注意してください。

独自の再帰関数でこれを行いたい場合は、次のようにします。

function recursive($needle, $array, $holder = []) 
{
    $holder = [];
    foreach($array as $key=>$value)
    {
       if($key===$needle && $value!=='')
       {
          $holder = array_merge($holder, [$value]);
       }
       if(is_array($value))
       {
          $holder = array_merge($holder, recursive($needle, $value, $holder));
       }
    }
    return $holder;
}
于 2013-10-31T14:47:40.860 に答える
1

インターフェイスを介した真の(tm) 再帰的配列トラバーサルRecursiveIteratorと、いくつかの主要なフィルターと配列変換関数を使用すると、よりきめ細かい制御が可能になる可能性があります。

$needle = '0';
$array  = [[1]];

$it = new KeyFilter(
    new RecursiveIteratorIterator(
        new MyRecursiveArrayIterator($array)
        , RecursiveIteratorIterator::SELF_FIRST
    )
    , $needle
);

$result = iterator_to_array($it, FALSE);
var_dump($result);

次のような模範的な結果を提供します。

array(2) {
  [0] =>
  array(1) {
    [0] =>
    int(1)
  }
  [1] =>
  int(1)
}

完全なコード例 ( Demo ):

<?php
/**
 * @link http://stackoverflow.com/q/19709410/367456
 */

Class MyRecursiveArrayIterator extends ArrayIterator implements RecursiveIterator
{
    public function hasChildren()
    {
        $current = $this->current();
        return is_array($current) && count($current);
    }

    public function getChildren()
    {
        return new self($this->current());
    }
}

class KeyFilter extends RegexIterator
{
    public function __construct(Iterator $iterator, $key)
    {
        parent::__construct(
            $iterator, '/' . preg_quote($key) . '/', NULL, RegexIterator::USE_KEY
        );
    }
}

$needle = '0';
$array  = [[1]];

$it = new KeyFilter(
    new RecursiveIteratorIterator(
        new MyRecursiveArrayIterator($array)
        , RecursiveIteratorIterator::SELF_FIRST
    )
    , $needle
);

$result = iterator_to_array($it, FALSE);
var_dump($result);
于 2013-10-31T15:15:36.557 に答える
0

あなたの構造の小さな変更:

$holder = recursive($needle, $value, $holder);

あ?

于 2013-10-31T14:58:26.180 に答える