
1. 问题背景与目标
在数据处理中,我们经常会遇到需要将一系列结构相似的扁平数据记录,按照某个共同的属性进行分组和组织。例如,一个从数据库查询结果中获取的数组,可能包含多条记录,每条记录都有一个object_type字段。我们的目标是将这些记录重新组织成一个多维数组,其中object_type作为顶级键,其值是一个包含所有具有该object_type的记录的子数组。
原始数据结构示例:
假设我们有以下PHP数组,其中包含多个事件记录,每条记录都包含initiator_id、object_type、object_id和date等信息。请注意,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' => 300,
'object_type' => 3,
'object_id' => 1001,
'date' => '2021-11-16 07:00:00',
],
];期望的目标数据结构:
立即学习“PHP免费学习笔记(深入)”;
我们希望将上述扁平数组转换为以下结构,其中object_type(例如1、2、3)作为主键,每个主键对应一个包含所有相关记录的子数组。
[
1 => [ // object_type 为 1 的所有记录
[ ... 记录1 ... ],
[ ... 记录2 ... ],
[ ... 记录3 ... ],
],
2 => [ // object_type 为 2 的所有记录
[ ... 记录4 ... ],
],
3 => [ // object_type 为 3 的所有记录
[ ... 记录5 ... ],
],
]2. 实现方法与代码示例
实现这种数组重构的核心思路是遍历原始数组,并根据指定键(object_type)的值动态构建新的多维数组。
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' => 300,
'object_type' => 3,
'object_id' => 1001,
'date' => '2021-11-16 07:00:00',
],
];
// 用于存储重构后的多维数组
$restructuredArray = [];
// 遍历原始数组中的每个子数组(即每条记录)
foreach ($originalArray as $item) {
// 获取当前记录的 object_type 值,作为新数组的主键
$objectType = $item['object_type'];
// 检查新数组中是否已经存在以当前 object_type 为键的子数组
// 如果不存在,则初始化一个空数组
if (!isset($restructuredArray[$objectType])) {
$restructuredArray[$objectType] = [];
}
// 将当前记录($item)添加到对应 object_type 的子数组中
$restructuredArray[$objectType][] = $item;
}
// 打印重构后的数组以验证结果
echo "";
print_r($restructuredArray);
echo "
";
?>3. 代码解析与输出结果
上述代码通过一个简单的foreach循环实现了数组的重构。
-
初始化 $restructuredArray: 创建一个空数组,用于存放最终重构后的数据。
-
遍历 $originalArray: 循环迭代原始数组中的每一个子数组(即每一条记录)。在每次迭代中,当前记录被赋值给 $item 变量。
-
获取分组键: 从 $item 中提取 object_type 的值,并将其存储在 $objectType 变量中。这个值将作为我们新多维数组的顶级键。
-
条件性初始化子数组: if (!isset($restructuredArray[$objectType])) 语句检查 $restructuredArray 中是否已经存在以当前 $objectType 为键的元素。如果不存在,这意味着这是我们第一次遇到这个 object_type,因此我们为它创建一个新的空数组 $restructuredArray[$objectType] = [];,以便后续添加记录。
-
添加记录到子数组: $restructuredArray[$objectType][] = $item; 将当前的 $item(即原始数组中的一条记录)添加到对应 $objectType 的子数组中。[] 语法确保了新记录作为子数组的一个新元素被追加,而不会覆盖已有的记录。
运行上述代码将得到以下输出:
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'
)
)
[3] => Array
(
[0] => Array
(
[initiator_id] => 300
[object_type] => 3
[object_id] => 1001
[date] => '2021-11-16 07:00:00'
)
)
)
4. 注意事项与总结
- 键的唯一性与数据类型: 用于分组的键(如object_type)的值应能作为有效的数组键。PHP数组键可以是整数或字符串。如果键值是其他类型(如对象或数组),PHP会尝试将其转换为字符串,这可能导致非预期的行为。
- 性能考量: 对于非常大的数据集,这种基于循环的重构方法是高效且直观的。在大多数Web应用场景中,其性能通常不是瓶颈。
- 灵活性: 这种模式非常灵活,可以轻松修改以根据其他键(如initiator_id或object_id)进行分组,只需更改$item['object_type']为相应的键即可。
- PHP 7.4+ 的简化语法(可选): 在PHP 7.4及更高版本中,可以使用短箭头函数和array_reduce或array_walk等函数结合更简洁的语法来实现类似功能,但对于初学者而言,foreach循环通常更易于理解和维护。
通过上述方法,我们可以有效地将扁平的数组数据按照业务逻辑需求进行结构化,使其更易于管理、访问和进一步处理。











