0

0

如何在嵌套树形数组中递归查找指定 slug 的节点及其完整子树

聖光之護

聖光之護

发布时间:2025-12-31 19:17:22

|

478人浏览过

|

来源于php中文网

原创

如何在嵌套树形数组中递归查找指定 slug 的节点及其完整子树

本文讲解如何使用递归函数在具有 `child` 键的多层嵌套数组(菜单树结构)中精准定位目标节点,并完整返回该节点及其全部后代,解决因递归调用未正确返回值导致结果为空的常见错误。

在处理菜单、分类或组织架构等树形数据时,常以嵌套数组形式存储,其中每个节点通过 'child' 键关联其子节点(如示例中的关联结构为 ['child']['2']['child']['4']...)。此时若需根据 slug 查找某节点并连同其整个子树一并返回,必须确保递归调用的结果被正确捕获与传递——而这正是原代码失败的核心原因。

原函数中虽检测到 child 存在,却注释掉了递归调用语句,且未对递归结果做任何判断或返回:

if (isset($row['child'])) {
    // return $this->getSlugData($slug, $row['child']); ← 被注释,且缺少 return
}

即使取消注释,也仅返回第一层子树的匹配结果,而忽略更深层的可能匹配;更重要的是,未检查递归调用是否真正找到了目标,导致即使子树中命中,父级循环仍继续执行并最终返回空数组 []。

✅ 正确做法是:

  • 对每一层级的 $row 先做直接匹配;
  • 若未命中,再对其所有子结构(不限于 'child' 键,而是遍历所有数组型子项)递归搜索;
  • 关键:每次递归调用后立即检查返回值,一旦非空即刻 return,避免被后续迭代覆盖。

以下是优化后的健壮实现:

唱鸭
唱鸭

音乐创作全流程的AI自动作曲工具,集 AI 辅助作词、AI 自动作曲、编曲、混音于一体

下载
function getSlugData(string $slug, array $data, string $key = 'slug'): array
{
    foreach ($data as $row) {
        // 1. 直接匹配当前节点
        if (isset($row[$key]) && $row[$key] === $slug) {
            return $row; // ✅ 找到即返回完整节点(含其 child)
        }

        // 2. 深度递归:遍历当前节点所有子数组(兼容 child、items、children 等不同键名)
        foreach ($row as $value) {
            if (is_array($value)) {
                $result = getSlugData($slug, $value, $key);
                if (!empty($result)) {
                    return $result; // ✅ 递归命中,立即返回,不继续循环
                }
            }
        }
    }

    return []; // ❌ 全部遍历完毕仍未找到
}
? 为什么不用硬编码 if (isset($row['child']))? 因为真实项目中树结构键名可能多样(如 children、items、submenus),甚至同一棵树混用多个键。上例采用“对每个数组值递归”的策略,天然支持任意嵌套结构,更具通用性与可维护性。

调用示例:

$tree = json_decode('[{"id":1,"slug":"medicinskaya-ge","child":{"2":{"id":2,"slug":"dnk-diagnostika","child":{"6":{"id":6,"slug":"testirovanie-ge"}}}}} ]', true);

var_dump(getSlugData('testirovanie-ge', $tree));
// 输出:包含 id=6 的完整节点,且保留其上级 child 结构链(即原始嵌套上下文)

var_dump(getSlugData('nonexistent', $tree)); // 输出:[]

? 注意事项总结:

  • 始终检查递归调用的返回值,避免“调用了却丢弃结果”;
  • 使用 !empty($result) 或 $result !== [] 判断,而非 isset($result)(因空数组也是合法返回);
  • 若需严格限定只查 'child' 键(如 API 协议强制),可将内层 foreach 替换为:
    if (isset($row['child']) && is_array($row['child'])) {
        $result = getSlugData($slug, $row['child'], $key);
        if (!empty($result)) return $result;
    }
  • 函数设计为纯函数(无副作用),便于单元测试与复用。

通过以上重构,即可稳定、高效地从任意深度的树形数组中提取目标节点及其完整子树。

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

712

2023.08.22

php中foreach用法
php中foreach用法

本专题整合了php中foreach用法的相关介绍,阅读专题下面的文章了解更多详细教程。

37

2025.12.04

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

4

2025.12.31

视频文件格式
视频文件格式

本专题整合了视频文件格式相关内容,阅读专题下面的文章了解更多详细内容。

7

2025.12.31

不受国内限制的浏览器大全
不受国内限制的浏览器大全

想找真正自由、无限制的上网体验?本合集精选2025年最开放、隐私强、访问无阻的浏览器App,涵盖Tor、Brave、Via、X浏览器、Mullvad等高自由度工具。支持自定义搜索引擎、广告拦截、隐身模式及全球网站无障碍访问,部分更具备防追踪、去谷歌化、双内核切换等高级功能。无论日常浏览、隐私保护还是突破地域限制,总有一款适合你!

7

2025.12.31

出现404解决方法大全
出现404解决方法大全

本专题整合了404错误解决方法大全,阅读专题下面的文章了解更多详细内容。

42

2025.12.31

html5怎么播放视频
html5怎么播放视频

想让网页流畅播放视频?本合集详解HTML5视频播放核心方法!涵盖<video>标签基础用法、多格式兼容(MP4/WebM/OGV)、自定义播放控件、响应式适配及常见浏览器兼容问题解决方案。无需插件,纯前端实现高清视频嵌入,助你快速打造现代化网页视频体验。

4

2025.12.31

关闭win10系统自动更新教程大全
关闭win10系统自动更新教程大全

本专题整合了关闭win10系统自动更新教程大全,阅读专题下面的文章了解更多详细内容。

3

2025.12.31

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.1万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

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

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