0

0

通用树中查找节点父节点:基于广度优先遍历的实现指南

花韻仙語

花韻仙語

发布时间:2025-08-11 17:24:20

|

1047人浏览过

|

来源于php中文网

原创

通用树中查找节点父节点:基于广度优先遍历的实现指南

本教程详细介绍了如何在通用树数据结构中查找指定节点的父节点。我们将采用广度优先遍历(BFS)算法,通过系统地逐层探索树的节点,高效地定位目标节点的父级。文章将涵盖节点结构定义、BFS算法原理、Java代码实现、时间与空间复杂度分析,以及相关注意事项,帮助读者掌握这一核心树操作技巧。

1. 通用树节点结构定义

首先,我们定义通用树的节点结构。一个通用树的节点通常包含一个键(key)用于标识,以及一个子节点列表(children),因为通用树的每个节点可以拥有任意数量的子节点。

import java.util.ArrayList;

public class Node {
    int key; // 节点键值
    ArrayList children = new ArrayList<>(); // 存储子节点的列表

    /**
     * 判断当前节点是否有子节点。
     * 虽然在查找父节点的逻辑中不直接使用,但有助于理解节点特性。
     * @return 如果有子节点返回 true,否则返回 false。
     */
    public boolean hasChildren(){
        return !children.isEmpty(); // 更简洁的判断方式
    }
}

2. 查找父节点:广度优先遍历(BFS)原理

要查找一个指定键值(token)的节点的父节点,我们可以利用广度优先遍历(Breadth-First Search, BFS)算法。BFS 是一种逐层探索树或图的算法,非常适合解决需要查找最近关系或最短路径的问题,例如查找父节点。

算法核心思想: BFS 使用队列(Queue)来管理待访问的节点。它从根节点开始,先访问当前节点的所有子节点,然后将这些子节点加入队列,再从队列中取出下一个节点进行同样的操作,直到队列为空或找到目标。

在查找父节点的场景中,当我们从队列中取出一个节点 p(作为潜在的父节点)时,我们会遍历它的所有子节点 c。如果发现某个子节点 c 的键值与我们正在寻找的 token 匹配,那么当前的 p 就是我们目标节点的父节点,我们可以立即返回 p。如果 c 不匹配,我们就将 c 加入队列,以便在后续的迭代中将其作为潜在的父节点进行检查。

Magician
Magician

Figma插件,AI生成图标、图片和UX文案

下载

3. Java 实现

下面是基于上述原理的 Java 实现代码:

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.Queue;

public class TreeOperations {

    /**
     * 在通用树中查找指定键值节点的父节点。
     * 使用广度优先遍历(BFS)实现。
     *
     * @param root 树的根节点。
     * @param token 要查找的子节点的键值。
     * @return 如果找到,返回目标节点的父节点;如果未找到或目标节点是根节点(无父节点),则返回 null。
     */
    public static Node findParent(Node root, int token) {
        // 如果树为空,或者根节点为空,直接返回null
        if (root == null) {
            return null;
        }

        // 使用LinkedList作为Queue的实现
        Queue queue = new LinkedList<>();
        queue.add(root); // 将根节点加入队列,作为遍历的起点

        // 广度优先遍历
        while (!queue.isEmpty()) {
            Node currentNode = queue.poll(); // 从队列中取出当前节点,它将作为潜在的父节点

            // 遍历当前节点的所有子节点
            for (Node child : currentNode.children) {
                // 如果子节点的键值与目标token匹配
                if (child.key == token) {
                    return currentNode; // 找到了,当前节点就是目标节点的父节点
                }
                // 如果子节点不匹配,将其加入队列,以便后续作为潜在的父节点进行检查
                queue.add(child);
            }
        }

        // 遍历完所有节点仍未找到,说明目标节点不存在或其是根节点(无父节点)
        return null;
    }

    public static void main(String[] args) {
        // 构建一个示例通用树
        //       1
        //      /|\
        //     2 3 4
        //    / \   \
        //   5   6   7
        Node root = new Node();
        root.key = 1;

        Node node2 = new Node(); node2.key = 2;
        Node node3 = new Node(); node3.key = 3;
        Node node4 = new Node(); node4.key = 4;
        root.children.add(node2);
        root.children.add(node3);
        root.children.add(node4);

        Node node5 = new Node(); node5.key = 5;
        Node node6 = new Node(); node6.key = 6;
        node2.children.add(node5);
        node2.children.add(node6);

        Node node7 = new Node(); node7.key = 7;
        node4.children.add(node7);

        // 测试查找
        Node parentOf6 = findParent(root, 6);
        if (parentOf6 != null) {
            System.out.println("节点 6 的父节点是: " + parentOf6.key); // 预期输出 2
        } else {
            System.out.println("未找到节点 6 的父节点。");
        }

        Node parentOf7 = findParent(root, 7);
        if (parentOf7 != null) {
            System.out.println("节点 7 的父节点是: " + parentOf7.key); // 预期输出 4
        } else {
            System.out.println("未找到节点 7 的父节点。");
        }

        Node parentOf1 = findParent(root, 1); // 根节点无父节点
        if (parentOf1 != null) {
            System.out.println("节点 1 的父节点是: " + parentOf1.key);
        } else {
            System.out.println("未找到节点 1 的父节点(或其是根节点)。"); // 预期输出 未找到...
        }

        Node parentOf99 = findParent(root, 99); // 不存在的节点
        if (parentOf99 != null) {
            System.out.println("节点 99 的父节点是: " + parentOf99.key);
        } else {
            System.out.println("未找到节点 99 的父节点。"); // 预期输出 未找到...
        }
    }
}

4. 复杂度分析

  • 时间复杂度: O(N),其中 N 是树中节点的总数。在最坏情况下,我们需要访问树中的所有节点才能找到目标节点的父节点(例如,目标节点是最后一个被访问的叶子节点,或者目标节点不存在)。每个节点最多被访问一次(入队一次,出队一次)。
  • 空间复杂度: O(W),其中 W 是树的最大宽度(即在任何一层中节点的最大数量)。在最坏情况下,例如一个完整的二叉树,最后一层的节点数量接近 N/2,此时队列中可能存储接近 N/2 个节点。对于一个高度为 H 的完美平衡树,空间复杂度为 O(N)。在极端情况下(例如,一个只有一层,包含所有子节点的扁平树),空间复杂度也是 O(N)。

5. 注意事项与总结

  • 根节点的父节点: 根节点没有父节点。如果 token 恰好是根节点的键值,findParent 函数将返回 null,这符合逻辑。
  • 目标节点不存在: 如果树中不存在与 token 匹配的节点,函数也将返回 null。
  • BFS的优势: 对于查找父节点这类需要遍历所有可能路径的问题,BFS 能够确保我们按层级顺序发现节点,并且在找到第一个匹配项时立即返回,效率较高。
  • 递归与迭代的选择: 尽管某些树遍历问题常采用递归(如深度优先遍历DFS),但对于查找父节点这类问题,广度优先遍历(BFS)通过迭代配合队列实现,通常更为直观和高效。当 currentNode 作为潜在父节点被处理时,其 child 如果匹配 token,那么 currentNode 立刻就是其父节点,这种层级关系在 BFS 中自然体现。而直接用递归(DFS)来查找父节点会更复杂一些,可能需要额外的参数来传递父节点信息或返回一个包含父子对的结果。

通过本教程,您应该已经掌握了在通用树中利用广度优先遍历查找指定节点父节点的方法。理解

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

236

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

458

2024.03.01

登录token无效
登录token无效

登录token无效解决方法:1、检查token的有效期限,如果token已经过期,需要重新获取一个新的token;2、检查token的签名,如果签名不正确,需要重新获取一个新的token;3、检查密钥的正确性,如果密钥不正确,需要重新获取一个新的token;4、使用HTTPS协议传输token,建议使用HTTPS协议进行传输 ;5、使用双因素认证,双因素认证可以提高账户的安全性。

6169

2023.09.14

登录token无效怎么办
登录token无效怎么办

登录token无效的解决办法有检查Token是否过期、检查Token是否正确、检查Token是否被篡改、检查Token是否与用户匹配、清除缓存或Cookie、检查网络连接和服务器状态、重新登录或请求新的Token、联系技术支持或开发人员等。本专题为大家提供token相关的文章、下载、课程内容,供大家免费下载体验。

818

2023.09.14

token怎么获取
token怎么获取

获取token值的方法:1、小程序调用“wx.login()”获取 临时登录凭证code,并回传到开发者服务器;2、开发者服务器以code换取,用户唯一标识openid和会话密钥“session_key”。想了解更详细的内容,可以阅读本专题下面的文章。

1066

2023.12.21

token什么意思
token什么意思

token是一种用于表示用户权限、记录交易信息、支付虚拟货币的数字货币。可以用来在特定的网络上进行交易,用来购买或出售特定的虚拟货币,也可以用来支付特定的服务费用。想了解更多token什么意思的相关内容可以访问本专题下面的文章。

1355

2024.03.01

treenode的用法
treenode的用法

​在计算机编程领域,TreeNode是一种常见的数据结构,通常用于构建树形结构。在不同的编程语言中,TreeNode可能有不同的实现方式和用法,通常用于表示树的节点信息。更多关于treenode相关问题详情请看本专题下面的文章。php中文网欢迎大家前来学习。

538

2023.12.01

C++ 高效算法与数据结构
C++ 高效算法与数据结构

本专题讲解 C++ 中常用算法与数据结构的实现与优化,涵盖排序算法(快速排序、归并排序)、查找算法、图算法、动态规划、贪心算法等,并结合实际案例分析如何选择最优算法来提高程序效率。通过深入理解数据结构(链表、树、堆、哈希表等),帮助开发者提升 在复杂应用中的算法设计与性能优化能力。

17

2025.12.22

clawdbot ai使用教程 保姆级clawdbot部署安装手册
clawdbot ai使用教程 保姆级clawdbot部署安装手册

Clawdbot是一个“有灵魂”的AI助手,可以帮用户清空收件箱、发送电子邮件、管理日历、办理航班值机等等,并且可以接入用户常用的任何聊天APP,所有的操作均可通过WhatsApp、Telegram等平台完成,用户只需通过对话,就能操控设备自动执行各类任务。

18

2026.01.29

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
10分钟--Midjourney创作自己的漫画
10分钟--Midjourney创作自己的漫画

共1课时 | 0.1万人学习

Midjourney 关键词系列整合
Midjourney 关键词系列整合

共13课时 | 0.9万人学习

AI绘画教程
AI绘画教程

共2课时 | 0.2万人学习

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

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