本文深入探讨了Java中通过树形结构找到特定类型部门的两种核心方法:递归和迭代。本文详细阐述了如何利用这两种策略有效地实现深度优先搜索,以解决复杂组织结构中类型筛选部门的实际问题,并提供了明确的代码示例和专业指导。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 递归或迭代也应用于遍历整棵树。
        // 一个更完整的 findDepartmentByPredicate 递归或迭代也应用于遍历整棵树。
        // 假设它能以某种方式获取所有部门并进行过滤,但在实际应用中需要重新实现深度遍历逻辑。
        return rootDepartments.stream() // 假设这里可以得到所有部门的流量
                .flatMap(dept -> {
                    // 这是一个简化的例子,实际上需要一种深度遍历和扁平化的方法
                    // 例如:getAllDepartmentsFlatStream().filter(predicate)
                    return dept.getMatchingDepartment(predicate).stream();
                });
    }
}

递归实现分析:

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

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

注意事项:

  • 递归深度过大可能导致栈溢出(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循环,只要栈不是空的,就会继续执行:
    • 作为currrent部门,从栈顶弹出一个Department。
    • 检查current部门的类型是否匹配,如果匹配,则添加到结果列表中。
    • 假如current部门是Company类型,那么它的所有子部门都可以获得。
    • 将这些子部门反向压入堆栈。这样做的目的是在下一个循环中弹出curent的第一个子部门,从而实现深度优先级的遍历顺序。

注意事项:

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

3. 总结

本文详细介绍了两种主要的深度优先搜索实现方法:递归和迭代,通过Java中的树形结构来搜索特定类型的部门。

  • 递归简单直观,易于理解,但可能面临栈溢出的风险。它通过辅助方法包装了遍历逻辑。
  • 迭代方法通过显式堆栈管理遍历状态,避免了递归深度的限制,更适合处理深度或动态变化的树形结构。它需要更精细地控制元素的进出顺序。

在实际开发中,选择哪种方法取决于具体的业务场景和性能、内存和代码可读性的平衡。对于大多数中等深度的树木来说,递归通常是一个更简单的选择;对于不可预测或需要严格控制内存的场景,迭代更稳定。

Logo

开源鸿蒙跨平台开发社区汇聚开发者与厂商,共建“一次开发,多端部署”的开源生态,致力于降低跨端开发门槛,推动万物智联创新。

更多推荐