使用Java Stream API获取子部门的所有父级部门

使用Java Stream API获取子部门的所有父级部门

过了腊八就是年
车票
订了吗?想家了吧.....

在数据库表中,经常会有包含层次结构的部门信息表。本博客将使用Java语言和Stream API演示如何获取给定子部门的所有父级部门,并在代码中加入详细注释。

1. 创建 Department 类

首先,我们需要定义一个类来表示部门信息。Department类包含三个属性:idnameparentId,分别表示部门的唯一标识、名称和父级部门的标识。

java 复制代码
public class Department {
    private long id;
    private String name;
    private long parentId;

    // 构造函数
    public Department(long id, String name, long parentId) {
        this.id = id;
        this.name = name;
        this.parentId = parentId;
    }

    // Getter和Setter方法省略
}

2. 编写 DepartmentUtils 类

接下来,我们创建一个DepartmentUtils类,其中包含一个静态方法getParentDepartments,该方法使用递归调用来获取给定子部门的所有父级部门。注释将详细解释每个关键步骤。

java 复制代码
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import java.util.Optional;
import java.util.stream.Collectors;
import java.util.stream.Stream;

public class DepartmentUtils {

    /**
     * 获取给定子部门的所有父级部门
     *
     * @param allDepartments 包含所有部门信息的列表
     * @param childDepartmentId 给定子部门的ID
     * @return 包含所有父级部门的列表
     */
    public static List<Department> getParentDepartments(List<Department> allDepartments, long childDepartmentId) {
        // 使用递归调用获取所有父级部门
        return getParentDepartmentsRecursively(allDepartments, childDepartmentId)
                .stream()
                .filter(Objects::nonNull)
                .collect(Collectors.toList());
    }

    /**
     * 递归方法:获取给定子部门的所有父级部门
     *
     * @param allDepartments 包含所有部门信息的列表
     * @param childDepartmentId 给定子部门的ID
     * @return 包含所有父级部门的列表
     */
    private static List<Department> getParentDepartmentsRecursively(List<Department> allDepartments, long childDepartmentId) {
        // 查找给定ID的子部门
        Optional<Department> childDepartment = allDepartments.stream()
                .filter(d -> d.getId() == childDepartmentId)
                .findFirst();

        if (childDepartment.isPresent()) {
            // 查找子部门的父部门
            Department parentDepartment = allDepartments.stream()
                    .filter(d -> Objects.equals(d.getId(), childDepartment.get().getParentId()))
                    .findFirst()
                    .orElse(null);

            if (parentDepartment != null) {
                // 递归调用,获取父部门的所有父级部门
                List<Department> parentDepartments = getParentDepartmentsRecursively(allDepartments, parentDepartment.getId());

                // 合并当前父部门和其所有父级部门
                return Stream.concat(Stream.of(parentDepartment), parentDepartments.stream())
                        .collect(Collectors.toList());
            }
        }

        // 如果子部门或其父部门未找到,返回空列表
        return new ArrayList<>();
    }
}

3. 使用示例

最后,我们创建一个使用示例,演示如何调用DepartmentUtils类来获取子部门的所有父级部门。这个示例包含了一个包含部门信息的列表,并打印了结果。

java 复制代码
import java.util.Arrays;
import java.util.List;

public class Main {
    public static void main(String[] args) {
        // 创建部门信息列表
        List<Department> allDepartments = Arrays.asList(
                new Department(1, "部门A", 0),
                new Department(2, "部门B", 1),
                new Department(3, "部门C", 2),
                // 添加其他部门信息
        );

        // 选择一个子部门的ID进行查询
        long childDepartmentId = 3;

        // 获取子部门的所有父级部门
        List<Department> parentDepartments = DepartmentUtils.getParentDepartments(allDepartments, childDepartmentId);

        // 打印结果
        System.out.println("子部门的所有父级部门:");
        parentDepartments.forEach(department -> System.out.println(department.getName()));
    }
}

4. 避免 NullPointerException

在代码中,我们对可能导致NullPointerException的情况进行了处理,确保在访问属性之前检查了null值。如果子部门或其父部门未找到,我们返回一个空列表而不是null,以避免潜在的异常。

相关推荐
火车叼位7 小时前
Windows 双网关自动切换:Node.js + 计划任务实现旁路由优先
网络协议·程序员
Spider Cat 蜘蛛猫7 小时前
Springboot SSO系统设计文档
java·spring boot·后端
嵌入式小企鹅8 小时前
UiPath推出AI编程“总指挥台”,SiFive发布RISC-V第三代猛兽
人工智能·学习·google·程序员·ai编程·risc-v·开源工具
zyk_computer9 小时前
AI 时代,或许 Rust 比 Python 更合适
人工智能·后端·python·ai·rust·ai编程·vibe coding
雨辰AI9 小时前
SpringBoot3 项目国产化改造完整流程|从 MySQL 到人大金仓落地
java·数据库·后端·mysql·政务
GreenTea10 小时前
【Rust 2026教程:从零构建 Mini-OLAP 引擎】第 6 章 Benchmark 与优化路线图
后端
Rust语言中文社区11 小时前
【Rust日报】2026-05-14 Pyrefly v1.0 正式发布:快速的 Python 类型检查器和语言服务器
开发语言·后端·python·rust
GreenTea11 小时前
【Rust 2026教程:从零构建 Mini-OLAP 引擎】第 5 章 SQL → 逻辑计划 → 物理计划
后端
GreenTea11 小时前
【Rust 2026教程:从零构建 Mini-OLAP 引擎】第 4 章 哈希聚合:GROUP BY 的核心
后端
IT_陈寒11 小时前
Vue的v-for为什么不加key也能工作?我差点翻车
前端·人工智能·后端