0

0

PHP多维数组重构:按指定键值分组数据

碧海醫心

碧海醫心

发布时间:2025-10-06 12:06:21

|

984人浏览过

|

来源于php中文网

原创

php多维数组重构:按指定键值分组数据

本文将详细介绍如何在PHP中将扁平化的关联数组列表重构为多维数组,核心思路是根据数组中某个特定键(例如 object_type)的值进行分组,将具有相同键值的所有子数组归集到同一个父级键下,从而实现数据的层次化组织,提高数据的可读性和管理效率。

引言:数据重构的需求

在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 ... ],
    ],
]

实现方法:遍历与分组

实现这种数据重构最直接且常用的方法是遍历原始数组,并根据指定键的值动态地构建新的多维数组。

示例代码

 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 "
";
print_r($groupedArray);
echo "
"; ?>

代码解析

  1. $groupedArray = [];: 我们首先创建一个空的 $groupedArray,这将是最终存储重构后数据的容器。
  2. foreach ($originalArray as $item): 遍历原始的扁平数组 $originalArray,每次循环 $item 变量将持有当前的一个关联数组(即一条记录)。
  3. if (isset($item['object_type'])): 这是一个健壮性检查,确保当前 $item 确实包含我们用于分组的 object_type 键。在实际应用中,如果可以保证该键总是存在,这行可以省略。
  4. $objectType = $item['object_type'];: 提取当前记录的 object_type 值,这个值将作为新多维数组的顶级键。
  5. if (!isset($groupedArray[$objectType])) { $groupedArray[$objectType] = []; }: 这是分组逻辑的关键。在向 $groupedArray 添加数据之前,我们检查是否已经存在以当前 $objectType 为键的子数组。如果不存在,就创建一个新的空数组,以便后续将记录添加到其中。
  6. $groupedArray[$objectType][] = $item;: 将当前的 $item(即原始记录)追加到 $groupedArray 中对应 $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
                )

            [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 分组的多维数组结构。

知了zKnown
知了zKnown

知了zKnown:致力于信息降噪 / 阅读提效的个人知识助手。

下载

注意事项与最佳实践

  • 键值类型: 用于分组的键 (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开发中非常实用,能够帮助开发者更好地组织和管理复杂数据,为后续的数据处理和展示打下坚实的基础。掌握这一技巧,将使您在处理各种数据结构转换时游刃有余。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
数据分析工具有哪些
数据分析工具有哪些

数据分析工具有Excel、SQL、Python、R、Tableau、Power BI、SAS、SPSS和MATLAB等。详细介绍:1、Excel,具有强大的计算和数据处理功能;2、SQL,可以进行数据查询、过滤、排序、聚合等操作;3、Python,拥有丰富的数据分析库;4、R,拥有丰富的统计分析库和图形库;5、Tableau,提供了直观易用的用户界面等等。

727

2023.10.12

SQL中distinct的用法
SQL中distinct的用法

SQL中distinct的语法是“SELECT DISTINCT column1, column2,...,FROM table_name;”。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

328

2023.10.27

SQL中months_between使用方法
SQL中months_between使用方法

在SQL中,MONTHS_BETWEEN 是一个常见的函数,用于计算两个日期之间的月份差。想了解更多SQL的相关内容,可以阅读本专题下面的文章。

350

2024.02.23

SQL出现5120错误解决方法
SQL出现5120错误解决方法

SQL Server错误5120是由于没有足够的权限来访问或操作指定的数据库或文件引起的。想了解更多sql错误的相关内容,可以阅读本专题下面的文章。

1243

2024.03.06

sql procedure语法错误解决方法
sql procedure语法错误解决方法

sql procedure语法错误解决办法:1、仔细检查错误消息;2、检查语法规则;3、检查括号和引号;4、检查变量和参数;5、检查关键字和函数;6、逐步调试;7、参考文档和示例。想了解更多语法错误的相关内容,可以阅读本专题下面的文章。

360

2024.03.06

oracle数据库运行sql方法
oracle数据库运行sql方法

运行sql步骤包括:打开sql plus工具并连接到数据库。在提示符下输入sql语句。按enter键运行该语句。查看结果,错误消息或退出sql plus。想了解更多oracle数据库的相关内容,可以阅读本专题下面的文章。

821

2024.04.07

sql中where的含义
sql中where的含义

sql中where子句用于从表中过滤数据,它基于指定条件选择特定的行。想了解更多where的相关内容,可以阅读本专题下面的文章。

581

2024.04.29

sql中删除表的语句是什么
sql中删除表的语句是什么

sql中用于删除表的语句是drop table。语法为drop table table_name;该语句将永久删除指定表的表和数据。想了解更多sql的相关内容,可以阅读本专题下面的文章。

423

2024.04.29

Golang 网络安全与加密实战
Golang 网络安全与加密实战

本专题系统讲解 Golang 在网络安全与加密技术中的应用,包括对称加密与非对称加密(AES、RSA)、哈希与数字签名、JWT身份认证、SSL/TLS 安全通信、常见网络攻击防范(如SQL注入、XSS、CSRF)及其防护措施。通过实战案例,帮助学习者掌握 如何使用 Go 语言保障网络通信的安全性,保护用户数据与隐私。

0

2026.01.29

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PHP课程
PHP课程

共137课时 | 10万人学习

JavaScript ES5基础线上课程教学
JavaScript ES5基础线上课程教学

共6课时 | 11.2万人学习

PHP新手语法线上课程教学
PHP新手语法线上课程教学

共13课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号