全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

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

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

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

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

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

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

✅ 正确做法是:

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

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

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;
    }
  • 函数设计为纯函数(无副作用),便于单元测试与复用。

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


# js  # json  # 编码  # 递归函数  # 为什么  # 架构  # if  # foreach  # 递归  # 循环  # 重构  # 子树  # 遍历  # 返回值  # 的是  # 若需  # 多个  # 键名  # 掉了  # 更重要 


相关文章: 专业制作网站的公司哪家好,建立一个公司网站的费用.有哪些部分,分别要多少钱?  建站之星伪静态规则如何设置?  重庆网站制作公司哪家好,重庆中考招生办官方网站?  如何选择适配移动端的WAP自助建站平台?  企业网站制作费用多少,企业网站空间一般需要多大,费用是多少?  建站之星与建站宝盒如何选择最佳方案?  测试制作网站有哪些,测试性取向的权威测试或者网站?  建站VPS配置与SEO优化指南:关键词排名提升策略  公司网站建设制作费用,想建设一个属于自己的企业网站,该如何去做?  定制建站模板如何实现SEO优化与智能系统配置?18字教程  深圳网站制作平台,深圳市做网站好的公司有哪些?  山东云建站价格为何差异显著?  如何在腾讯云服务器上快速搭建个人网站?  成都网站制作价格表,现在成都广电的单独网络宽带有多少的,资费是什么情况呢?  武汉网站如何制作,黄黄高铁武穴北站途经哪些村庄?  如何用花生壳三步快速搭建专属网站?  制作国外网站的软件,国外有哪些比较优质的网站推荐?  如何快速重置建站主机并恢复默认配置?  如何快速查询网址的建站时间与历史轨迹?  建站org新手必看:2024最新搭建流程与模板选择技巧  建站主机是否等同于虚拟主机?  简历在线制作网站免费版,如何创建个人简历?  如何优化Golang Web性能_Golang HTTP服务器性能提升方法  如何快速搭建虚拟主机网站?新手必看指南  如何在Windows环境下新建FTP站点并设置权限?  详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)  Android自定义控件实现温度旋转按钮效果  做企业网站制作流程,企业网站制作基本流程有哪些?  如何选择高性价比服务器搭建个人网站?  免费制作小说封面的网站有哪些,怎么接网站批量的封面单?  如何获取上海专业网站定制建站电话?  C#如何在一个XML文件中查找并替换文本内容  如何在Tomcat中配置并部署网站项目?  ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?  油猴 教程,油猴搜脚本为什么会网页无法显示?  如何配置IIS站点权限与局域网访问?  建站之星如何保障用户数据免受黑客入侵?  用v-html解决Vue.js渲染中html标签不被解析的问题  网站专业制作公司有哪些,做一个公司网站要多少钱?  内部网站制作流程,如何建立公司内部网站?  网站好制作吗知乎,网站开发好学吗?有什么技巧?  代购小票制作网站有哪些,购物小票的简要说明?  专业商城网站制作公司有哪些,pi商城官网是哪个?  较简单的网站制作软件有哪些,手机版网页制作用什么软件?  个人网站制作流程图片大全,个人网站如何注销?  深圳网站制作费用多少钱,读秀,深圳文献港这样的网站很多只提供网上试读,但有些人只要提供试读的文章就能全篇下载,这个是怎么弄的?  建站之星安装后界面空白如何解决?  c# 服务器GC和工作站GC的区别和设置  潮流网站制作头像软件下载,适合母子的网名有哪些?  实现虚拟支付需哪些建站技术支撑? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。