4

我从一个任意深度的嵌套数组开始。在该数组中,一些键是一系列由点分隔的标记。例如“billingAddress.street”或“foo.bar.baz”。我想将这些键元素扩展为数组,因此结果是一个嵌套数组,所有这些键都展开了。

例如:

[
    'billingAddress.street' => 'My Street',
    'foo.bar.baz' => 'biz',
]

应扩展为:

[
    'billingAddress' => [
        'street' => 'My Street',
    ],
    'foo' => [
        'bar' => [
            'baz' => 'biz',
        ]
    ]
]

原始的“billingAddress.street”可以留在新的“billingAddress”数组旁边,但不是必须的(因此解决方案可以对原始数组进行操作或创建新数组)。可能需要将其他元素(例如“billingAddress.city”)添加到数组的同一扩展部分。

一些键可能有两个以上由点分隔的标记,因此需要更深入地扩展。

我看过array_walk_recursive()但只对元素起作用。对于每个匹配的元素键,我实际上想修改这些元素所在的父数组。

我看过array_map,但它不提供对密钥的访问,而且据我所知不是递归的。

要扩展的示例数组:

[
    'name' => 'Name',
    'address.city' => 'City',
    'address.street' => 'Street',
    'card' => [
        'type' => 'visa',
        'details.last4' => '1234',
    ],
]

这将扩展为:

[
    'name' => 'Name',
    'address.city' => 'City', // Optional
    'address' => [
        'city' => 'City',
        'street' => 'Street',
    ],
    'address.street' => 'Street', // Optional
    'card' => [
        'type' => 'visa',
        'details.last4' => '1234', // Optional
        'details' => [
            'last4' => '1234',
        ],
    ],
]

我认为我需要的是遍历array嵌套数组中的每个并且可以对其应用用户函数的东西。但我确实怀疑我遗漏了一些明显的东西。我正在使用的支付网关使用点符号向我发送这种数组和“假装数组”的组合,我的目标是将其规范化为一个数组以提取部分。

我相信这个问题与 SO 上的类似问题不同,因为这种数组和非数组的混合用于扩展。从概念上讲,它是一个嵌套数组,其中任何级别的合理元素都需要用新数组替换,因此这里发生了两个级别的递归:树行走,扩展,然后扩展树的行走以查看如果需要更多的扩展。

4

3 回答 3

5

您会发现反转通过组合(虚线)键爆炸获得的键的顺序很有用。以这种相反的顺序,更容易将先前的结果逐步包装到新数组中,从而为一个点分键/值对创建嵌套结果。

最后,可以使用内置array_merge_recursive函数将该部分结果合并到累积的“大”结果中:

function expandKeys($arr) {
    $result = [];
    foreach($arr as $key => $value) {
        if (is_array($value)) $value = expandKeys($value);
        foreach(array_reverse(explode(".", $key)) as $key) $value = [$key => $value];
        $result = array_merge_recursive($result, $value);
    }
    return $result;
}

看到它在repl.it上运行

于 2018-07-28T18:08:33.743 回答
4

这是一个递归尝试。请注意,这不会删除旧键,不会维护任何键顺序并忽略 type 的键foo.bar.baz

function expand(&$data) {
  if (is_array($data)) {
    foreach ($data as $k => $v) {
      $e = explode(".", $k);

      if (count($e) == 2) {
        [$a, $b] = $e;
        $data[$a][$b]= $v;
      }

      expand($data[$k]);
    }
  }
}

结果:

Array
(
    [name] => Name
    [address.city] => City
    [address.street] => Street
    [card] => Array
        (
            [type] => visa
            [details.last4] => 1234
            [details] => Array
                (
                    [last4] => 1234
                )

        )

    [address] => Array
        (
            [city] => City
            [street] => Street
        )

)

解释:

在函数的任何调用中,如果参数是一个数组,则遍历键和值以查找其中带有 a 的键.。对于任何此类键,将它们展开。对数组中的所有键递归调用此函数。

完整版本:

这是一个支持多个.s 并在之后清理密钥的完整版本:

function expand(&$data) {
  if (is_array($data)) {
    foreach ($data as $k => $v) {
      $e = explode(".", $k);
      $a = array_shift($e);

      if (count($e) == 1) {
        $data[$a][$e[0]] = $v;
      }
      else if (count($e) > 1) {
        $data[$a][implode(".", $e)] = $v;
      }
    }

    foreach ($data as $k => $v) {
      expand($data[$k]);

      if (preg_match('/\./', $k)) {
        unset($data[$k]);
      }
    }
  }
}
于 2018-07-28T16:53:29.453 回答
2

@trincot 的另一个解决方案被认为更优雅,并且是我现在使用的解决方案。

这是我的解决方案,它扩展了@ggorlen 给出的解决方案和提示

我采取的方法是:

  • 创建一个新数组而不是对初始数组进行操作。
  • 无需保留旧的预扩展元素。如果需要,可以轻松添加它们。
  • 扩展键是从根数组一次完成一层,其余的扩展以递归方式传回。

类方法:

protected function expandKeys($arr)
{
    $result = [];

    while (count($arr)) {
        // Shift the first element off the array - both key and value.
        // We are treating this like a stack of elements to work through,
        // and some new elements may be added to the stack as we go.

        $value = reset($arr);
        $key = key($arr);
        unset($arr[$key]);

        if (strpos($key, '.') !== false) {
            list($base, $ext) = explode('.', $key, 2);

            if (! array_key_exists($base, $arr)) {
                // This will be another array element on the end of the
                // arr stack, to recurse into.

                $arr[$base] = [];
            }

            // Add the value nested one level in.
            // Value at $arr['bar.baz.biz'] is now at $arr['bar']['baz.biz']
            // We may also add to this element before we get to processing it,
            // for example $arr['bar.baz.bam']


            $arr[$base][$ext] = $value;
        } elseif (is_array($value)) {
            // We already have an array value, so give the value
            // the same treatment in case any keys need expanding further.

            $result[$key] = $this->expandKeys($value);
        } else {
            // A scalar value with no expandable key.

            $result[$key] = $value;
        }
    }

    return $result;
}

$result = $this->expandKeys($sourceArray)
于 2018-07-28T17:57:45.870 回答