0

0

Java 树形结构深度搜索与部门查找实践

聖光之護

聖光之護

发布时间:2025-09-30 10:34:14

|

265人浏览过

|

来源于php中文网

原创

Java 树形结构深度搜索与部门查找实践

本文深入探讨了在Java中遍历树形结构以查找特定类型部门的两种核心方法:递归与迭代。通过定义Department和Company接口构建树状层级,文章详细阐述了如何利用这两种策略高效地实现深度优先搜索,从而解决在复杂组织结构中按类型筛选部门的实际问题,并提供了清晰的代码示例与专业指导。

1. 理解树形结构与业务场景

在企业组织管理中,部门通常具有层级关系,例如一个公司下设多个部门,而某些部门本身也可能包含子部门,这自然形成了一个树形结构。为了在java中表示这种结构并对其进行操作,我们可以定义如下接口:

import java.util.List;
import java.util.Optional;
import java.util.function.Predicate;

/**
 * 部门接口,定义了部门的基本属性和行为。
 */
public interface Department {
    String getName(); // 获取部门名称
    String getType(); // 获取部门类型

    /**
     * 默认方法:检查当前部门是否匹配给定谓词。
     * 此方法仅检查当前节点,不涉及子部门的遍历。
     * @param predicate 用于测试部门的条件
     * @return 如果匹配则返回包含当前部门的Optional,否则返回空的Optional
     */
    default Optional<Department> getMatchingDepartment(Predicate<Department> predicate) {
        if (predicate.test(this)) {
            return Optional.of(this);
        }
        return Optional.empty();
    }
}
/**
 * 公司接口,继承自Department,并表示一个可以包含子部门的实体。
 * 它构成了树形结构中的内部节点。
 */
public interface Company extends Department {
    List<Department> getDepartments(); // 获取当前公司下的所有子部门

    /**
     * 默认方法:尝试查找当前公司直接子部门中第一个匹配谓词的部门。
     * 注意:此方法仅遍历直接子部门,不进行深度递归。
     * @param predicate 用于测试部门的条件
     * @return 如果找到匹配的直接子部门则返回包含该部门的Optional,否则返回空的Optional
     */
    default Optional<Department> getMatchingDepartment(Predicate<Department> predicate) {
        // 此实现仅遍历直接子部门,不深入到子部门的子部门
        return getDepartments().stream()
                .map(department -> department.getMatchingDepartment(predicate))
                .filter(Optional::isPresent)
                .map(Optional::get)
                .findFirst();
    }
}

我们的目标是实现一个功能,能够根据给定的部门类型,从整个公司层级结构中找出所有匹配的部门。

2. 深度优先搜索(DFS)的挑战与解决方案

在上述接口中,Company接口的getMatchingDepartment默认实现只能遍历其直接子部门,无法深入到子部门的子部门,因此无法实现整个树的深度搜索。要解决这个问题,我们需要采用深度优先搜索(DFS)策略,这通常可以通过递归或迭代两种方式实现。

2.1 递归式深度优先搜索

递归是实现树形结构深度遍历最直观的方式之一。其核心思想是:对于当前节点,首先检查自身是否符合条件;如果当前节点是一个可以包含子节点的类型(例如Company),则对其所有子节点递归调用相同的查找逻辑。

import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.function.Predicate;
import java.util.stream.Stream;

/**
 * Concern 类用于封装部门查找逻辑。
 * 假设 Concern 内部持有一个根部门列表,代表整个组织的顶层结构。
 */
public class Concern {
    private final List<Department> rootDepartments; // 假设这是整个组织的顶层部门列表

    public Concern(List<Department> rootDepartments) {
        this.rootDepartments = rootDepartments;
    }

    /**
     * 根据部门类型查找所有匹配的部门(递归实现)。
     *
     * @param type 要查找的部门类型
     * @return 匹配指定类型的所有部门列表
     */
    public List<Department> findDepartmentByType(String type) {
        ArrayList<Department> result = new ArrayList<>();
        // 从根部门开始递归查找
        findDepartmentByTypeRecursiveImpl(type, rootDepartments, result);
        return result;
    }

    /**
     * 递归辅助方法:深度遍历部门树,查找指定类型的部门。
     *
     * @param type        要查找的部门类型
     * @param departments 当前需要遍历的部门列表
     * @param result      用于收集匹配部门的列表
     */
    private void findDepartmentByTypeRecursiveImpl(String type, List<Department> departments, List<Department> result) {
        if (departments == null || departments.isEmpty()) {
            return; // 递归终止条件:当前列表为空
        }

        for (Department current : departments) {
            // 1. 检查当前部门是否匹配类型
            if (type.equals(current.getType())) {
                result.add(current);
            }
            // 2. 如果当前部门是公司类型,则递归遍历其子部门
            if (current instanceof Company) {
                findDepartmentByTypeRecursiveImpl(type, ((Company) current).getDepartments(), result);
            }
        }
    }

    // 以下是原始问题中提供的其他查找方法,为保持上下文完整性保留
    public Optional<Department> findDepartmentByName(String name) {
        return findDepartmentByPredicate(department -> department.getName().equals(name)).findFirst();
    }

    private Stream<Department> findDepartmentByPredicate(Predicate<Department> predicate) {
        // 此处的实现需要改进,它依赖于 Department 和 Company 接口中的 getMatchingDepartment 默认方法,
        // 但这些默认方法在原始问题中未能实现深度遍历。
        // 一个更完整的 findDepartmentByPredicate 应该也使用递归或迭代来遍历整个树。
        // 为了演示,此处假设它能某种方式获取到所有部门并进行过滤,但实际应用中需要重新实现深度遍历逻辑。
        return rootDepartments.stream() // 假设这里能获取到所有部门的流
                .flatMap(dept -> {
                    // 这是一个简化的示例,实际需要一个深度遍历并扁平化的方法
                    // 例如:getAllDepartmentsFlatStream().filter(predicate)
                    return dept.getMatchingDepartment(predicate).stream();
                });
    }
}

递归实现解析:

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

  • findDepartmentByType(String type) 是公共入口方法,它初始化一个结果列表并调用私有的辅助方法 findDepartmentByTypeRecursiveImpl。
  • findDepartmentByTypeRecursiveImpl 是递归的核心。它遍历当前部门列表:
    • 对于每个current部门,首先检查其类型是否与目标type匹配。如果匹配,则将其添加到结果列表中。
    • 接着,通过 instanceof Company 判断current是否是一个Company(即一个内部节点,可能包含子部门)。
    • 如果current是Company,则递归调用 findDepartmentByTypeRecursiveImpl,传入该Company的子部门列表,继续向下遍历。
  • 递归的终止条件是当传入的部门列表为空时,或者当遍历到叶子节点(非Company类型的Department)时,递归路径自然结束。

注意事项:

百宝箱
百宝箱

百宝箱是支付宝推出的一站式AI原生应用开发平台,无需任何代码基础,只需三步即可完成AI应用的创建与发布。

下载
  • 递归深度过大可能导致溢出(StackOverflowError),尤其是在处理非常深的树形结构时。
  • 需要一个辅助方法来隐藏递归的实现细节,保持公共接口的简洁。

2.2 迭代式深度优先搜索(使用栈)

迭代方式实现深度优先搜索通常使用一个显式的栈(Stack)数据结构来模拟递归调用的过程。这种方法可以避免递归深度过大导致的栈溢出问题。

import java.util.ArrayList;
import java.util.Collections;
import java.util.List;
import java.util.Stack; // Java 传统的 Stack 类,也可以使用 ArrayDeque

// ... Concern 类的其他部分保持不变 ...

public class Concern {
    private final List<Department> rootDepartments;

    public Concern(List<Department> rootDepartments) {
        this.rootDepartments = rootDepartments;
    }

    /**
     * 根据部门类型查找所有匹配的部门(迭代实现)。
     *
     * @param type 要查找的部门类型
     * @return 匹配指定类型的所有部门列表
     */
    public List<Department> findDepartmentByTypeIterative(String type) {
        ArrayList<Department> result = new ArrayList<>();
        Stack<Department> stack = new Stack<>(); // 使用 Stack 模拟递归栈

        // 将所有根部门压入栈中,作为遍历的起点
        // 注意:这里需要确保按照正确的顺序入栈,以模拟DFS的遍历顺序
        // 如果希望从左到右遍历,通常需要反向入栈
        for (int i = rootDepartments.size() - 1; i >= 0; i--) {
            stack.push(rootDepartments.get(i));
        }
        // 或者更简单地,直接添加所有,但要注意后续处理顺序
        // stack.addAll(rootDepartments); // 如果使用 ArrayList 作为栈,添加到末尾,然后从末尾移除

        while (!stack.isEmpty()) {
            Department current = stack.pop(); // 弹出栈顶部门进行处理

            // 1. 检查当前部门是否匹配类型
            if (type.equals(current.getType())) {
                result.add(current);
            }

            // 2. 如果当前部门是公司类型,将其子部门压入栈中
            if (current instanceof Company) {
                List<Department> children = ((Company) current).getDepartments();
                // 将子部门反向压入栈中,以确保在下次迭代时,按照从左到右的顺序处理子部门
                // 或者说,为了保持DFS的特性,后入栈的先处理,所以需要反向入栈
                for (int i = children.size() - 1; i >= 0; i--) {
                    stack.push(children.get(i));
                }
            }
        }
        return result;
    }

    // ... 其他方法 ...
}

迭代实现解析:

  • findDepartmentByTypeIterative(String type) 是公共入口方法。
  • 它初始化一个ArrayList作为结果列表,并创建一个Stack(或java.util.ArrayDeque作为更推荐的栈实现)来存储待访问的部门。
  • 首先,将所有根部门压入栈中。为了模拟DFS的从左到右遍历(如果子节点有顺序),通常需要反向压入子节点。
  • 进入while循环,只要栈不为空,就持续执行:
    • 从栈顶弹出一个Department作为current部门。
    • 检查current部门的类型是否匹配,如果匹配则添加到结果列表。
    • 如果current部门是Company类型,则获取其所有子部门。
    • 将这些子部门反向压入栈中。这样做的目的是,当下次循环弹出时,会先弹出current的第一个子部门,从而实现深度优先的遍历顺序。

注意事项:

  • 迭代方式避免了递归深度限制,但在极端情况下,显式栈可能会占用较多内存。
  • 使用java.util.ArrayDeque作为栈通常比java.util.Stack性能更好,因为它基于数组实现,避免了同步开销。
  • 入栈和出栈的顺序对于遍历的顺序(前序、中序、后序)至关重要。此处实现的是前序遍历(先处理当前节点,再处理子节点)。

3. 总结

本文详细介绍了在Java中遍历树形结构以查找特定类型部门的两种主要深度优先搜索实现方式:递归和迭代。

  • 递归方式简洁直观,易于理解,但可能面临栈溢出的风险。它通过一个辅助方法实现,将遍历逻辑封装起来。
  • 迭代方式通过显式使用栈来管理遍历状态,避免了递归深度限制,更适合处理非常深或动态变化的树形结构。它需要更精细地控制元素的入栈和出栈顺序。

在实际开发中,选择哪种方法取决于具体的业务场景和对性能、内存以及代码可读性的权衡。对于大多数中等深度的树,递归通常是更简洁的选择;而对于深度不可预测或需要严格控制内存使用的场景,迭代方式则更为稳健。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

106

2023.09.25

treenode的用法
treenode的用法

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

549

2023.12.01

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

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

30

2025.12.22

深入理解算法:高效算法与数据结构专题
深入理解算法:高效算法与数据结构专题

本专题专注于算法与数据结构的核心概念,适合想深入理解并提升编程能力的开发者。专题内容包括常见数据结构的实现与应用,如数组、链表、栈、队列、哈希表、树、图等;以及高效的排序算法、搜索算法、动态规划等经典算法。通过详细的讲解与复杂度分析,帮助开发者不仅能熟练运用这些基础知识,还能在实际编程中优化性能,提高代码的执行效率。本专题适合准备面试的开发者,也适合希望提高算法思维的编程爱好者。

44

2026.01.06

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1925

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

656

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2392

2025.12.29

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

76

2026.03.11

热门下载

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

精品课程

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

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.2万人学习

Java 教程
Java 教程

共578课时 | 81万人学习

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

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