将树形数组扁平化为列表数组的PHP教程

霞舞
发布: 2025-10-12 13:15:00
原创
934人浏览过

将树形数组扁平化为列表数组的PHP教程

本教程详细介绍了如何使用php将复杂的树形数组结构高效地扁平化为简洁的列表数组。通过递归遍历树的每个节点,提取关键数据并移除层级信息,最终生成一个按id排序的、易于处理的扁平化数组。文章提供了清晰的代码示例和注意事项,帮助开发者理解并实现数据结构的转换。

在PHP开发中,我们经常会遇到将数据组织成树形结构(例如菜单、评论回复、组织架构等)的需求。然而,在某些场景下,例如将数据存储到数据库、进行全局搜索或需要线性处理数据时,我们又需要将这种树形结构“反向”扁平化回一个简单的列表数组。本文将深入探讨如何高效地实现这一转换过程。

数据结构概览

为了更好地理解扁平化过程,我们首先来看一下涉及的两种数据结构:

1. 原始扁平数组(构建树的源数据)

这是一个典型的扁平数组,每个元素包含 id、parent_id 和 name 字段,通过 parent_id 字段建立了层级关系。parent_id 为 0 的节点通常表示根节点。

立即学习PHP免费学习笔记(深入)”;

$sourceArray = [
    ['id' => 1, 'parent_id' => 0, 'name' => 'id1'],
    ['id' => 2, 'parent_id' => 1, 'name' => 'id2'],
    ['id' => 3, 'parent_id' => 1, 'name' => 'id3'],
    ['id' => 4, 'parent_id' => 0, 'name' => 'id4'],
    ['id' => 5, 'parent_id' => 2, 'name' => 'id5'],
    ['id' => 6, 'parent_id' => 3, 'name' => 'id6'],
    ['id' => 7, 'parent_id' => 0, 'name' => 'id7'],
    ['id' => 8, 'parent_id' => 3, 'name' => 'id8'],
    ['id' => 9, 'parent_id' => 4, 'name' => 'id9'],
    ['id' => 10, 'parent_id' => 9, 'name' => 'id10'],
];
登录后复制

2. 待扁平化的树形数组

这是由上述扁平数组通过某种构建树的函数转换而来的树形结构。它通过 children 键来表示子节点,形成了嵌套的层级关系。需要注意的是,根节点通常没有 parent_id 字段,其 parent_id 在扁平化后应被视为 0。

$tree = [
    [
        "id" => 1,
        "name" => "id1",
        "children" => [
            [
                "id" => 2,
                "parent_id" => 1,
                "name" => "id2",
                "children" => [
                    ["id" => 5, "parent_id" => 2, "name" => "id5"]
                ]
            ],
            [
                "id" => 3,
                "parent_id" => 1,
                "name" => "id3",
                "children" => [
                    ["id" => 6, "parent_id" => 3, "name" => "id6"],
                    ["id" => 8, "parent_id" => 3, "name" => "id8"]
                ]
            ]
        ]
    ],
    [
        "id" => 4,
        "name" => "id4",
        "children" => [
            [
                "id" => 9,
                "parent_id" => 4,
                "name" => "id9",
                "children" => [
                    ["id" => 10, "parent_id" => 9, "name" => "id10"]
                ]
            ]
        ]
    ],
    ["id" => 7, "name" => "id7", "children" => []]
];
登录后复制

3. 期望的扁平化结果数组

扁平化后的目标是将上述树形结构还原成类似于 sourceArray 的形式,但通常会按照 id 进行排序,并且确保每个节点都包含 id、parent_id 和 name 字段。

即构数智人
即构数智人

即构数智人是由即构科技推出的AI虚拟数字人视频创作平台,支持数字人形象定制、短视频创作、数字人直播等。

即构数智人 36
查看详情 即构数智人
// 期望的输出(按ID排序)
[
    ['id' => 1, 'parent_id' => 0, 'name' => 'id1'],
    ['id' => 2, 'parent_id' => 1, 'name' => 'id2'],
    ['id' => 3, 'parent_id' => 1, 'name' => 'id3'],
    ['id' => 4, 'parent_id' => 0, 'name' => 'id4'],
    ['id' => 5, 'parent_id' => 2, 'name' => 'id5'],
    ['id' => 6, 'parent_id' => 3, 'name' => 'id6'],
    ['id' => 7, 'parent_id' => 0, 'name' => 'id7'],
    ['id' => 8, 'parent_id' => 3, 'name' => 'id8'],
    ['id' => 9, 'parent_id' => 4, 'name' => 'id9'],
    ['id' => 10, 'parent_id' => 9, 'name' => 'id10'],
]
登录后复制

核心实现:递归扁平化函数

将树形结构扁平化的核心思想是深度优先遍历(DFS)。我们通过递归函数遍历树的每一个节点,提取其核心数据(id, name, parent_id),并将其添加到最终的扁平化列表中。children 键在提取数据时将被忽略,因为它只用于表示层级关系。

flattenTreeNodes 函数详解

此函数是递归处理树形结构的关键。它接收当前层级的节点数组和通过引用传递的扁平化结果数组。

/**
 * 递归地将树形节点扁平化为列表。
 *
 * @param array $treeNodes 当前层级的树形节点数组。
 * @param array $flatList 引用传递,用于累积扁平化后的节点列表。
 * @return array 扁平化后的节点列表。
 */
function flattenTreeNodes(array $treeNodes, array &$flatList = []): array
{
    foreach ($treeNodes as $node) {
        // 提取当前节点的核心数据
        // 使用 ?? 运算符确保 parent_id 存在,如果不存在则默认为 0 (适用于根节点)
        $nodeData = [
            'id' => $node['id'],
            'parent_id' => $node['parent_id'] ?? 0,
            'name' => $node['name'],
        ];

        // 将当前节点数据添加到扁平化列表中
        $flatList[] = $nodeData;

        // 如果当前节点有子节点,则递归调用自身处理子节点
        if (isset($node['children']) && is_array($node['children']) && !empty($node['children'])) {
            flattenTreeNodes($node['children'], $flatList);
        }
    }
    return $flatList;
}
登录后复制

函数逻辑说明:

  1. 遍历当前层级节点: 函数通过 foreach 循环遍历 $treeNodes 数组中的每一个节点。
  2. 提取节点数据: 对于每个 $node,我们创建一个新的 $nodeData 数组,只包含 id、parent_id 和 name 字段。
    • parent_id 的处理:这里使用了 PHP 7+ 的 null 合并运算符 ??。如果 $node['parent_id'] 存在,则使用其值;否则,默认为 0。这巧妙地处理了树形结构中根节点可能没有 parent_id 字段的情况。
  3. 添加到扁平列表: 将 $nodeData 添加到通过引用传递的 $flatList 数组中。由于 $flatList 是引用,所有递归调用都会向同一个数组中添加数据。
  4. 递归处理子节点: 检查当前节点是否包含 children 键,并且 children 是一个非空数组。如果是,则递归调用 flattenTreeNodes 函数,将当前节点的 children 数组作为新的 $treeNodes 参数,继续向下遍历。

辅助函数:排序与重置索引

为了使最终的扁平化数组更规范和易于使用,我们通常会对其进行排序(例如按 id 升序),并重置数组的数字索引。

getFlatArrayFromTree 函数详解

这个函数作为入口点,负责调用递归函数并对结果进行后处理。

/**
 * 将完整的树形结构扁平化为按ID排序的列表数组。
 *
 * @param array $tree 完整的树形结构数组。
 * @return array 扁平化并排序后的列表数组。
 */
function getFlatArrayFromTree(array $tree): array
{
    $flatArray = [];
    // 调用递归函数进行扁平化
    flattenTreeNodes($tree, $flatArray);

    // 对扁平化后的数组按 'id' 字段进行升序排序
    usort($flatArray, static function (array $node1, array $node2) {
        // PHP 7+ 的飞船运算符 (<=>) 简洁地实现三向比较
        return ($node1['id'] <=> $node2['id']);
    });

    // 重置数组的数字键,确保从 0 开始连续索引
    return array_values($flatArray);
}
登录后复制

函数逻辑说明:

  1. 初始化: 创建一个空的 $flatArray 数组,用于接收扁平化后的数据。
  2. 调用扁平化: 调用 flattenTreeNodes 函数,传入 $tree 和 $flatArray。flattenTreeNodes 会将所有扁平化后的节点添加到 $flatArray 中。
  3. 排序: 使用 usort 函数对 $flatArray 进行自定义排序。匿名函数定义了排序规则:根据每个节点的 id 字段进行升序排列。<=> 运算符(飞船运算符)在 PHP 7+ 中非常方便,它会返回 -1、0 或 1,分别表示小于、等于或大于。
  4. 重置索引: array_values 函数会返回数组中所有值的新数组,并重新建立数字索引(从 0 开始)。这确保了即使在排序过程中原始键被打乱,最终结果也是一个连续索引的数组。

完整示例与运行

将上述函数与示例数据结合,可以得到一个完整的可运行代码:

<?php

/**
 * 递归地将树形节点扁平化为列表。
 *
 * @param array $treeNodes 当前层级的树形节点数组。
 * @param array $flatList 引用传递,用于累积扁平化后的节点列表。
 * @return array 扁平化后的节点列表。
 */
function flattenTreeNodes(array $treeNodes, array &$flatList = []): array
{
    foreach ($treeNodes as $node) {
        // 提取当前节点的核心数据
        // 使用 ?? 运算符确保 parent_id 存在,如果不存在则默认为 0 (适用于根节点)
        $nodeData = [
            'id' => $node['id'],
            'parent_id' => $node['parent_id'] ?? 0,
            'name' => $node['name'],
        ];

        // 将当前节点数据添加到扁平化列表中
        $flatList[] = $nodeData;

        // 如果当前节点有子节点,则递归调用自身处理子节点
        if (isset($node['children']) && is_array($node['children']) && !empty($node['children'])) {
            flattenTreeNodes($node['children'], $flatList);
        }
    }
    return $flatList;
}

/**
 * 将完整的树形结构扁平化为按ID排序的列表数组。
 *
 * @param array $tree 完整的树形结构数组。
 * @return array 扁平化并排序后的列表数组。
 */
function getFlatArrayFromTree(array $tree): array
{
    $flatArray = [];
    // 调用递归函数进行扁平化
    flattenTreeNodes($tree, $flatArray);

    // 对扁平化后的数组按 'id' 字段进行升序排序
登录后复制

以上就是将树形数组扁平化为列表数组的PHP教程的详细内容,更多请关注php中文网其它相关文章!

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

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