Java Stream流轻松遍历树形结构

在日常开发中,经常会遇到返回树形及遍历树形结构的需求,比如构建菜单,项目-任务树形结构,角色权限树形结构等。数据库一般使用parent_id来表示上一层级,遍历的时候一般使用递归或多层for循环,增加了数据库的查询压力。

为了降低数据库的查询压力,我们可以使用Java8中的Stream流一次性把数据查出来,然后通过流式处理。

话不多说,直接上Demo。

首先,创建一个Menu 菜单的实体类


@Data
@Builder
public class Menu {/*** 唯一id*/public Integer id;/*** 菜单名称*/public String name;/*** 上级id ,根节点为0*/public Integer parentId;/*** 子节点信息*/public List<Menu> childList;public Menu(Integer id, String name, Integer parentId) {this.id = id;this.name = name;this.parentId = parentId;}public Menu(Integer id, String name, Integer parentId, List<Menu> childList) {this.id = id;this.name = name;this.parentId = parentId;this.childList = childList;}
}

接下来,模拟生成一颗树形结构List<Menu> 集合。

private static List<Menu> createTree() {//模拟创建一棵树return  = Arrays.asList(new Menu(1, "目录", 0),new Menu(2, "第一章节", 1),new Menu(3, "1.1 大战光明顶", 2),new Menu(4, "1.2 偶遇赵敏", 2),new Menu(5, "1.3 迎娶周芷若", 2),new Menu(6, "第二章节", 1),new Menu(7, "2.1 挥泪斩狮王", 6),new Menu(8, "2.2 大意失街亭", 6),new Menu(9, "2.2.1 败走麦城", 7),new Menu(10, "2.2.2 火烧赤壁", 7),new Menu(11, "第三章节", 1),new Menu(12, "3.1 刺杀董卓", 11),new Menu(13, "3.1.1 貂蝉献艺", 12),new Menu(14, "结束", 0));}

然后,使用根据上级 parentId 获取所有子节点的方法,这里我们可以使用递归的方式去实现

private static List<Menu> getAllChildrenByRoot(Menu root, List<Menu> all) {return all.stream().filter(m -> Objects.equals(m.getParentId(), root.getId())).peek((m) -> m.setChildList(getAllChildrenByRoot(m, all))).collect(Collectors.toList());}

最后,使用main方法验证一下

public static void main(String[] args) {List<Menu> menus = createTree();List<Menu> menuList = menus.stream().filter(m -> m.getParentId() == 0).peek((m) -> m.setChildList(getAllChildrenByRoot(m, menus))).collect(Collectors.toList());System.out.println("/*****collect to json*****/");System.out.println(JSONObject.toJSON(menuList));}

打印一下结果

[{"name":"目录","childList":[{"name":"第一章节","childList":[{"name":"1.1 大战光明顶","childList":[],"id":3,"parentId":2},{"name":"1.2 偶遇赵敏","childList":[],"id":4,"parentId":2},{"name":"1.3 迎娶周芷若","childList":[],"id":5,"parentId":2}],"id":2,"parentId":1},{"name":"第二章节","childList":[{"name":"2.1 挥泪斩狮王","childList":[{"name":"2.2.1 败走麦城","childList":[],"id":9,"parentId":7},{"name":"2.2.2 火烧赤壁","childList":[],"id":10,"parentId":7}],"id":7,"parentId":6},{"name":"2.2 大意失街亭","childList":[],"id":8,"parentId":6}],"id":6,"parentId":1},{"name":"第三章节","childList":[{"name":"3.1 刺杀董卓","childList":[{"name":"3.1.1 貂蝉献艺","childList":[],"id":13,"parentId":12}],"id":12,"parentId":11}],"id":11,"parentId":1}],"id":1,"parentId":0},{"name":"结束","childList":[],"id":14,"parentId":0}
]

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/583625.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

计算机网络复习4

网络层——点到点 文章目录 网络层——点到点功能路由算法IPV4NAT 网络地址转换子网划分与子网掩码、CIDR地址解析协议ARP&#xff1a;根据IP地址找到MAC地址动态主机配置协议DHCP网际控制报文协议ICMPIPV6内部网关协议&#xff08;IGP&#xff09;外部网关协议(EGP) 功能 异构…

数字人直播系统——打破时空限制的新媒体时代

随着科技的不断进步&#xff0c;新媒体开始逐渐成为人们获取信息和娱乐的首选方式。其中&#xff0c;数字人直播系统作为一种创新的传媒形式&#xff0c;正以其独特的优势受到越来越多人的关注和喜爱。数字人直播系统通过将虚拟人物与现实世界紧密结合&#xff0c;打破了时间和…

SuperMap YashanDB联合解决方案发布,赋能更强大的地理智慧

近期&#xff0c;深圳计算科学研究院&#xff08;简称“深算院”&#xff09;携手超图软件集团&#xff08;简称“超图 ”&#xff09;重磅推出基于崖山数据库的空间数据管理解决方案&#xff0c;基于YashanDB空间数据库能力&#xff0c;与超图SuperMap GIS平台深度适配&#x…

【温故而知新】vue运用之探讨下单页面应用(SPA)与多页面应用(MPA)

一、概念 1.单页面应用SPA(Single page application) Vue单页面应用是一种采用Vue.js框架开发的Web应用程序,它仅有一个HTML文件,通过前端路由实现页面的切换和渲染。与传统的多页面应用相比,Vue单页面应用在用户体验和开发效率方面有着明显的优势。 在Vue单页面应用中…

计算机与人工智能:共创智能时代的新篇章

计算机与人工智能&#xff1a;共创智能时代的新篇章 在这个科技日新月异的时代&#xff0c;计算机与人工智能&#xff08;AI&#xff09;的结合正以前所未有的速度改变着世界。它们在各自的领域内飞速发展&#xff0c;而当这两者相遇时&#xff0c;它们产生了巨大的能量&#x…

C++ Qt开发:TableView与TreeView组件联动

Qt 是一个跨平台C图形界面开发库&#xff0c;利用Qt可以快速开发跨平台窗体应用程序&#xff0c;在Qt中我们可以通过拖拽的方式将不同组件放到指定的位置&#xff0c;实现图形化开发极大的方便了开发效率&#xff0c;本章将重点介绍TableView与TreeView组件联动的常用方法及灵活…

== 和 equals() 的区别

大家好&#xff0c;我是"java继父"伯约&#xff0c;这篇对大家有帮助的话求一个赞&#xff0c;另外文章末尾放了我从月入7k到现在3W的学习资料&#xff0c;大家可以去领一下&#xff08;无偿&#xff09;。 对于基本类型和引用类型的作用效果是不同的&#xff1a; …

再谈动态SQL

专栏精选 引入Mybatis Mybatis的快速入门 Mybatis的增删改查扩展功能说明 mapper映射的参数和结果 Mybatis复杂类型的结果映射 Mybatis基于注解的结果映射 Mybatis枚举类型处理和类型处理器 文章目录 专栏精选摘要引言正文动态sql标签ifchoose...when...otherwisewhere、…

【AI】阿里云免费GPU服务资源领取方法

首先&#xff0c;直接点击链接&#xff1a;阿里云免费试用 也可以复制链接到浏览器进行跳转&#xff1a;https://free.aliyun.com?userCodernbj0c1o 页面如下所示&#xff1a;这里的免费试用期限是3个月&#xff0c;给的资源点够我们试用V100 16G显存服务器300个小时&#xff…

作业--day38

1.定义一个Person类&#xff0c;包含私有成员&#xff0c;int *age&#xff0c;string &name&#xff0c;一个Stu类&#xff0c;包含私有成员double *score&#xff0c;Person p1&#xff0c;写出Person类和Stu类的特殊成员函数&#xff0c;并写一个Stu的show函数&#xff…

智慧工地云平台源码 支持二次开发、支持源码交付

智慧工地利用移动互联、物联网、云计算、大数据等新一代信息技术&#xff0c;彻底改变传统施工现场各参建方的交互方式、工作方式和管理模式&#xff0c;为建设集团、施工企业、监理单位、设计单位、政府监管部门等提供一揽子工地现场管理信息化解决方案。 通过人员管理、车辆管…

自己写的Js有序的Map, 并且支持Key为对象

var map newMap(); map.put({name:"张三"}, 23); map.put("李四", 24); map.get({name:"张三"}); map.remove({name:"张三"}); map.put("张三", 23); var objArray map.allList("name", "age&…

linux find 与cp 配合使用

cp与find配合使用 一、用自带的-exec参数 find ab -iname "ef" -type f -exec cp {} tmp \;-type f 文件 d 目录 可执行文件 -type f -executable {} 是查找结果 后面空格\; 结束 有点不太好用 但后面还可以 加命令 分号隔开 $find ab -name "ef" -…

SQLServer性能分析

目录 1 常用系统存储过程2 查看执行计划3 查看磁盘使用率&#xff08;STATISTICS IO&#xff09;4 查询时间耗时较长的语句5 死锁处理参考资料 1 常用系统存储过程 1.1 查询表结构 EXEC sp_columns your_table, column_name table_column1.2 查询表索引 EXEC sp_helpindex …

平时的一些思考内容

文章目录 阶乘位运算求概率 阶乘 阶乘是一很迷人的&#xff0c;刚开始的的变化还不是很大&#xff0c;到后面变化类似于直线上升的&#xff0c;不知道现实中哪些实例来表示阶乘。19的阶乘就已经超过了long了&#xff0c;在竞赛或者其他中要求2023或者很大数字的阶乘就需要考虑…

还在用Jekins?快来试试这款比Jekins简而轻的自动部署软件!

大家好&#xff0c;我是 Java陈序员。 在工作中&#xff0c;你是否遇到过团队中没有专业的运维&#xff0c;开发还要做运维的活&#xff0c;需要自己手动构建、部署项目&#xff1f; 不同的项目还有不同的部署命令&#xff0c;需要使用 SSH 工具连接远程服务器和使用 FTP 文件…

开源大语言模型简记

文章目录 开源大模型LlamaChinese-LLaMA-AlpacaLlama2-ChineseLinlyYaYiChatGLMtransformersGPT-3(未完全开源)BERTT5QwenBELLEMossBaichuan其他

web前端之JavaScript

MENU JavaScript之设计模式、单例、代理、装饰者、中介者、观察者、发布订阅、策略JavaScript之数组静态方法的实现、reduce、forEach、map、push、every JavaScript之设计模式、单例、代理、装饰者、中介者、观察者、发布订阅、策略 单例模式 概念 保证一个类仅有一个实例&am…

C基础使用

return 0; 语句用于表示退出程序。 一个程序有且只能有一个main函数的存在 安装编译环境&#xff1a; 安装vim: ubuntu里vim编辑器使用方法_ubuntu vim-CSDN博客 编译与运行&#xff1a; gcc hello.c //编译源文件 ./a.out //运行…

SpringBoot 项目中常用的注解

每一层对应每个包&#xff0c;包名中应全为小写。 一、Common 层&#xff08;实体类&#xff09; 前提&#xff1a;导入 Lombok 依赖 Data&#xff1a;生成 get 和 set 方法以及 toString 方法 Getter&#xff1a;只生成 get 方法&#xff0c;避免对类中的成员变量修改。 …