2

我有一个深度多维数组,我需要它来提取特定键的值。我发现该array_walk_recursive功能将是我最好的选择。我只需要第一次出现。

我的数组看起来像这样 - (除了更复杂)

Array (
    [vehicle info] => Array (
        [one] => Array (
            [submodel] => LX
            [engine] => 2.3
        )
        [two] => Array (
            [color] => blue
            [year] => 2007
            [wheels] => 4
        )
        [three] => Array (
            [submodel] => LX
            [make] => Ford
            [model] => F-150
            [offroad] => No
        )
    )
)

这里的问题是,submodel在第一和第三。此外,数组并不一致,所以我必须使用array_walk_recursive它来搜索匹配的键,然后返回该键的值。

这是我当前的代码 -

array_walk_recursive ($array, (function ($item, $key) {
    $wanted = "submodel";
    if ($key === $wanted) {
        echo ("$key is $item");
    }
}));

以上返回submodel is LXsubmodel is LX

奖金问题! 如何搜索多个键并为每个键返回第一个对应值?我在想把所有想要的键放在一个数组中,然后做一个 foreach 循环,但不太知道如何构造它。我是 php 新手。

4

4 回答 4

4

array_walk_recursive()是调用此任务的适当本机函数。跟踪结果数组中已经声明了哪些键,并确保它们永远不会被覆盖。

代码:(演示

$needles = ['submodel', 'offroad'];
$result = [];
array_walk_recursive(
    $array,
    function($value, $key) use ($needles, &$result) {
        if (
            in_array($key, $needles)
            && !isset($result[$key])
        ) {
            $result[$key] = "$key is $value";
        }
    }
);
var_export($result);

输出:

array (
  'submodel' => 'submodel is LX',
  'offroad' => 'offroad is No',
)

如果您的应用程序有性能问题,那么本机函数就会变得不那么有吸引力,因为它总是会迭代整个输入数组的结构——即使在遇到所有寻找的键之后也是如此。如果您想“提前中断”(短路),那么您将需要设计自己的递归函数,该函数将在找到所有寻找的键时返回。

代码:(演示

$soughtKeys = array_flip(['submodel', 'offroad']);

function earlyReturningRecursion(array $array, array $soughtKeys, array &$result = []): array
{
    foreach ($array as $key => $value) {
        if (!array_diff_key($soughtKeys, $result)) {  // check if result is complete
            return $result;
        } elseif (is_array($value)) {
            earlyReturningRecursion($value, $soughtKeys, $result);
        } elseif (isset($soughtKeys[$key]) && !isset($result[$key])) {
            $result[$key] = "$key is $value";
        }
    }
    return $result;
}
var_export(earlyReturningRecursion($array, $soughtKeys));
// same output as the first snippet
于 2017-06-05T20:18:50.997 回答
2

我将首先设置您想要的值null,然后仅在尚未找到它们时才保存它们,方法是检查is_null(). 我没有测试过这段代码,但它应该看起来像这样:

$submodel = null;
array_walk_recursive ($array, (function ($item, $key) {
    $wanted = "submodel";
    if ($key === $wanted && is_null($submodel)) {
        echo ("$key is $item");
        $submodel = $item;
    }
}));
于 2017-06-05T19:48:09.643 回答
2

array_walk_recursive() 具有不允许返回匹配结果的缺陷,但是在 PHP 7 中,您可以使用匿名函数和变量来存储匹配值。

$matching = null;
$wanted = "submodel";

array_walk_recursive ($array, function ($item, $key) use ($wanted, $matching) {
    if (($key === $wanted) && is_null($matching)) {
        $matching = $item;
    }
});
于 2017-06-05T19:57:22.917 回答
2

至于没有办法提前返回array_walk_recursive(),我建议创建一个函数来查找第一次出现的$wanted

$arr = [
  'vehicle info' => [
     'one' => ['submodel' => 'LX', 'engine' => '2.3'],
     'two' => ['color' => 'blue', 'year' => '2007', 'wheels' => '4'],
     'three' => ['submodel' => 'LX', 'make' => 'Ford', 'model' => 'F-150', 'offroad' => 'No'],
    ],
];

function find($needle, $haystack, $found = '')
{
    foreach ($haystack as $key => $value) {
        if ($found) {
            break;
        }
        if ($key === $needle) {
            $found = "{$needle} is {$value}";
            break;
        }
        if (is_array($value)) {
            $found = find($needle, $value, $found);
        }
    }
    return $found;
}

$wanted = 'submodel';
$result = find($wanted, $arr);

var_dump($result); // string(14) "submodel is LX"

现场演示


更新:要搜索多个键,您需要循环执行:

$multiple_keys = array('submodel', 'year');

foreach ($multiple_keys as $wanted) {
    var_dump(find($wanted, $arr));
}

// Output:
//    string(14) "submodel is LX"
//    string(12) "year is 2007"

现场演示

于 2017-06-05T20:18:27.983 回答