
在PHP开发中,我们经常会遇到需要处理从数据库查询或其他数据源获取的扁平化数组数据。这些数据通常是一系列关联数组的列表,每个关联数组代表一个独立的记录。然而,在某些场景下,为了更好地组织和管理数据,或者为了满足特定的业务逻辑和前端展示需求,我们需要将这些扁平数据重构为更具层次感的多维数组。一个常见的需求是根据某个共同的属性(如 object_type)将所有相关的记录分组到一起。
考虑以下原始数据结构,这是一个包含多个记录的数组,其中 object_type 字段可能重复:
$originalArray = [
[
'initiator_id' => 259,
'object_type' => 1,
'object_id' => 905,
'date' => '2021-11-16 06:24:16',
],
[
'initiator_id' => 259,
'object_type' => 1,
'object_id' => 905,
'date' => '2021-11-16 04:54:54',
],
[
'initiator_id' => 259,
'object_type' => 1,
'object_id' => 905,
'date' => '2021-11-16 04:53:58',
],
[
'initiator_id' => 219,
'object_type' => 2,
'object_id' => 915,
'date' => '2021-11-16 04:53:58',
],
[
'initiator_id' => 220,
'object_type' => 3,
'object_id' => 916,
'date' => '2021-11-16 04:53:58',
],
[
'initiator_id' => 221,
'object_type' => 2,
'object_id' => 917,
'date' => '2021-11-16 04:53:58',
],
];我们的目标是将这个数组转换为一个多维数组,其中 object_type 的值将作为新的顶级键,每个顶级键下包含一个数组,该数组中存储所有 object_type 相同的原始记录。
经过重构后,我们期望得到的数据结构如下所示:
立即学习“PHP免费学习笔记(深入)”;
[
1 => [ // object_type = 1 的所有记录
[ ... 原始记录0 ... ],
[ ... 原始记录1 ... ],
[ ... 原始记录2 ... ],
],
2 => [ // object_type = 2 的所有记录
[ ... 原始记录3 ... ],
[ ... 原始记录5 ... ],
],
3 => [ // object_type = 3 的所有记录
[ ... 原始记录4 ... ],
],
]实现这种数据重构最直接且常用的方法是遍历原始数组,并根据指定键的值动态地构建新的多维数组。
<?php
$originalArray = [
[
'initiator_id' => 259,
'object_type' => 1,
'object_id' => 905,
'date' => '2021-11-16 06:24:16',
],
[
'initiator_id' => 259,
'object_type' => 1,
'object_id' => 905,
'date' => '2021-11-16 04:54:54',
],
[
'initiator_id' => 259,
'object_type' => 1,
'object_id' => 905,
'date' => '2021-11-16 04:53:58',
],
[
'initiator_id' => 219,
'object_type' => 2,
'object_id' => 915,
'date' => '2021-11-16 04:53:58',
],
[
'initiator_id' => 220,
'object_type' => 3,
'object_id' => 916,
'date' => '2021-11-16 04:53:58',
],
[
'initiator_id' => 221,
'object_type' => 2,
'object_id' => 917,
'date' => '2021-11-16 04:53:58',
],
];
$groupedArray = []; // 初始化一个空数组用于存放重构后的数据
foreach ($originalArray as $item) {
// 检查当前元素是否包含 'object_type' 键
if (isset($item['object_type'])) {
$objectType = $item['object_type'];
// 如果 $groupedArray 中还没有以当前 objectType 为键的数组,则先创建一个
if (!isset($groupedArray[$objectType])) {
$groupedArray[$objectType] = [];
}
// 将当前元素添加到对应 objectType 的数组中
$groupedArray[$objectType][] = $item;
}
}
echo "<pre>";
print_r($groupedArray);
echo "</pre>";
?>执行上述代码,将得到以下输出:
Array
(
[1] => Array
(
[0] => Array
(
[initiator_id] => 259
[object_type] => 1
[object_id] => 905
[date] => 2021-11-16 06:24:16
)
[1] => Array
(
[initiator_id] => 259
[object_type] => 1
[object_id] => 905
[date] => 2021-11-16 04:54:54
)
[2] => Array
(
[initiator_id] => 259
[object_type] => 1
[object_id] => 905
[date] => 2021-11-16 04:53:58
)
)
[2] => Array
(
[0] => Array
(
[initiator_id] => 219
[object_type] => 2
[object_id] => 915
[date] => 2021-11-16 04:53:58
)
[1] => Array
(
[initiator_id] => 221
[object_type] => 2
[object_id] => 917
[date] => '2021-11-16 04:53:58
)
)
[3] => Array
(
[0] => Array
(
[initiator_id] => 220
[object_type] => 3
[object_id] => 916
[date] => 2021-11-16 04:53:58
)
)
)这个输出完美地实现了我们预期的按 object_type 分组的多维数组结构。
键值类型: 用于分组的键 (object_type) 的值可以是字符串或整数。PHP 会自动处理这些作为数组键。
性能考量: 对于非常庞大的数组(例如数十万甚至上百万条记录),上述 foreach 循环的性能通常是可接受的。然而,如果遇到极端情况,可以考虑使用更底层的优化或数据库层面的分组(例如 SQL 的 GROUP BY 子句)来减少 PHP 脚本的内存和CPU开销。
健壮性: 在实际项目中,总是建议对用于分组的键进行 isset() 或 array_key_exists() 检查,以避免因数据不一致导致未定义索引错误。
通用性: 可以将上述逻辑封装成一个函数,使其更具通用性,接受原始数组和用于分组的键名作为参数:
function groupArrayByField(array $data, string $field): array
{
$grouped = [];
foreach ($data as $item) {
if (isset($item[$field])) {
$value = $item[$field];
if (!isset($grouped[$value])) {
$grouped[$value] = [];
}
$grouped[$value][] = $item;
}
}
return $grouped;
}
$groupedData = groupArrayByField($originalArray, 'object_type');
// print_r($groupedData);array_reduce 方法: 对于更函数式编程风格的开发者,也可以使用 array_reduce 来实现同样的功能,但对于初学者来说,foreach 循环通常更易读和理解。
$groupedArrayReduce = array_reduce($originalArray, function ($carry, $item) {
if (isset($item['object_type'])) {
$objectType = $item['object_type'];
$carry[$objectType][] = $item;
}
return $carry;
}, []);
// print_r($groupedArrayReduce);请注意,在使用 array_reduce 时,如果 $carry[$objectType] 不存在,PHP 会自动将其创建为数组。
通过简单的 foreach 循环和条件判断,我们能够高效且灵活地将扁平化的关联数组列表重构为按指定键值分组的多维数组。这种数据重构技术在PHP开发中非常实用,能够帮助开发者更好地组织和管理复杂数据,为后续的数据处理和展示打下坚实的基础。掌握这一技巧,将使您在处理各种数据结构转换时游刃有余。
以上就是PHP多维数组重构:按指定键值分组数据的详细内容,更多请关注php中文网其它相关文章!
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号