实现树形结构几种方式

 1,三级分类树形结构查询

/*** DDD(Domain-Driven Design): 领域驱动设计** 三级分类树形结构;* 支持无限层级;* 当前项目只有三级*/
@Data
public class CategoryTreeTo {private  Long categoryId; //1private String categoryName;private List<CategoryTreeTo>  categoryChild;//子分类
}
   <!--    定义Category树形封装规则; 三级固定封装--><resultMap id="CategoryTreeRM" type="com.azxc.rapid.modules.main.dto.CategoryTreeTo"><!--     一级分类的规则   --><id column="id" property="categoryId"></id><result column="name" property="categoryName"></result><collection property="categoryChild"ofType="com.azxc.rapid.modules.main.dto.CategoryTreeTo"><!--  二级分类封装规则 --><id column="c2_id" property="categoryId"></id><result column="c2_name" property="categoryName"></result><collection property="categoryChild"ofType="com.azxc.rapid.modules.main.dto.CategoryTreeTo"><!--  三级分类封装规则 --><id column="c3_id" property="categoryId"></id><result column="c3_name" property="categoryName"></result></collection></collection></resultMap><select id="getTest" resultMap="CategoryTreeRM">select bc1.*,bc2.id   c2_id,bc2.name c2_name,bc2.category1_id,bc3.id   c3_id,bc3.name c3_name,bc3.category2_idfrom base_category1 bc1left join base_category2 bc2 on bc1.id = bc2.category1_idleft join base_category3 bc3 on bc2.id = bc3.category2_id</select>

执行结果展示

 2,树形结构(递归遍历)

    @Overridepublic List<Permission> getTest1() {//查询所有菜单List<Permission> res = baseMapper.getTest1();return build(res);}/*** 根据权限列表构建父子关系** @param permissionList* @return*/public static List<Permission> build(List<Permission> permissionList) {List<Permission> menu = new ArrayList<>();for (Permission permission : permissionList) {//判断当前菜单是否是以及菜单if (permission.getParentId() == 0) {//一级菜单//2.1 设置一级菜单的子菜单列表permission.setChildren(getChildren(permission, permissionList));//2.2 将一级菜单添加到菜单列表中menu.add(permission);}}return menu;}/*** 从原始菜单中获取某个权限的子菜单列表** @param permission* @param originPermissionList* @return*/private static List<Permission> getChildren(Permission permission, List<Permission> originPermissionList) {//1. 创建一个新的集合,用来存储子菜单List<Permission> children = new ArrayList<>();//1. 遍历出原始菜单中的每一个权限for (Permission child : originPermissionList) {//1.1 如果originPermission的父id等于permission的idif (permission.getId().equals(child.getParentId())) {//originPermission是permission的子菜单,则将originPermission添加到children中// 子菜单还有没有子菜单呢?child.setChildren(getChildren(child, originPermissionList));children.add(child);}}return children;}

结果 :

 表结构:树形结构

3,数据字典

需求效果图

 表结构:树形结构

 1.xml

    <select id="findListByParentId" resultType="com.atguigu.entity.Dict"><include refid="columns"></include>from hse_dictwhere parent_id = #{parentId} and is_deleted = 0</select>

2.业务层

    @Overridepublic List<Map<String, Object>> findZnodes(Long id) {//根据id查询对应的子分类List<Dict> dictList = dictMapper.findListByParentId(id);//使用Stream API将dictList转换为Map集合List<Map<String, Object>> zNodes = dictList.stream().map(dict -> {Map<String, Object> responseMap = new HashMap<>();//表示当前节点是否还有子节点,以当前节点的id到hse_dict表中查询子节点的数量,如果大于0,则表示当前节点还有子节点Integer count = dictMapper.countIsParent(dict.getId());responseMap.put("isParent", count > 0);//表示当前节点的名称responseMap.put("name", dict.getName());responseMap.put("id", dict.getId());return responseMap;}).collect(Collectors.toList());return zNodes;}

4,stream流递归实现

实体类
public class TreeBean {/*** id*/private Integer id;/*** 名称*/private String name;/*** 父id ,根节点为0*/public Integer parentId;/*** 子节点信息*/public List<TreeBean> childList;public TreeBean() {}public TreeBean(Integer id, String name, Integer parentId, List<TreeBean> childList) {this.id = id;this.name = name;this.parentId = parentId;this.childList = childList;}public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}public Integer getParentId() {return parentId;}public void setParentId(Integer parentId) {this.parentId = parentId;}public List<TreeBean> getChildList() {return childList;}public void setChildList(List<TreeBean> childList) {this.childList = childList;}/*** 初始化数据* @return*/public List<TreeBean> initializationData() {List<TreeBean> list = new ArrayList<>();TreeBean t1 = new TreeBean(1, "广东省", 0, new ArrayList<>());TreeBean t2 = new TreeBean(2, "湖南省", 0, new ArrayList<>());TreeBean t3 = new TreeBean(3, "广州市", 1, new ArrayList<>());TreeBean t4 = new TreeBean(4, "长沙市", 2, new ArrayList<>());TreeBean t5 = new TreeBean(5, "白云区", 3, new ArrayList<>());TreeBean t6 = new TreeBean(6, "黄浦区", 3, new ArrayList<>());TreeBean t7 = new TreeBean(7, "白云街道", 5, new ArrayList<>());TreeBean t8 = new TreeBean(8, "深圳市", 1, new ArrayList<>());TreeBean t9 = new TreeBean(9, "宝安区", 8, new ArrayList<>());TreeBean t10 = new TreeBean(10, "福田区", 8, new ArrayList<>());TreeBean t11 = new TreeBean(11, "南山区", 8, new ArrayList<>());TreeBean t12 = new TreeBean(12, "南山街道", 11, new ArrayList<>());TreeBean t13 = new TreeBean(13, "芙蓉区", 4, new ArrayList<>());TreeBean t14 = new TreeBean(14, "岳麓区", 4, new ArrayList<>());TreeBean t15 = new TreeBean(15, "开福区", 4, new ArrayList<>());TreeBean t16 = new TreeBean(16, "岳阳市", 2, new ArrayList<>());TreeBean t17 = new TreeBean(17, "岳麓街道", 14, new ArrayList<>());list.add(t1);list.add(t2);list.add(t3);list.add(t4);list.add(t5);list.add(t6);list.add(t7);list.add(t8);list.add(t9);list.add(t10);list.add(t11);list.add(t12);list.add(t13);list.add(t14);list.add(t15);list.add(t16);list.add(t17);return list;}
}
实现类
/*** 方式一:Stream流递归实现遍历树形结构*/public static void main(String[] args) {//获取数据List<TreeBean> treeBeans = new TreeBean().initializationData();//获取父节点List<TreeBean> collect = treeBeans.stream().filter(t -> t.getParentId() == 0).map(m -> {m.setChildList(getChildren(m, treeBeans));return m;}).collect(Collectors.toList());System.out.println(JSON.toJSONString(collect));}/*** 递归查询子节点* @param root  根节点* @param all   所有节点* @return 根节点信息*/public static List<TreeBean> getChildren(TreeBean root, List<TreeBean> all) {List<TreeBean> children = all.stream().filter(t -> {return Objects.equals(t.getParentId(), root.getId());}).map(m -> {m.setChildList(getChildren(m, all));return m;}).collect(Collectors.toList());return children;}
结果

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

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

相关文章

操作系统入门系列-MIT6.828(操作系统工程)学习笔记(四)---- C语言与计算机架构(Programming xv6 in C)

系列文章目录 操作系统入门系列-MIT6.S081&#xff08;操作系统&#xff09;学习笔记&#xff08;一&#xff09;---- 操作系统介绍与接口示例 操作系统入门系列-MIT6.828&#xff08;操作系统工程&#xff09;学习笔记&#xff08;二&#xff09;----课程实验环境搭建&#x…

linux系统——route路由命令

route路由对linux内的ip路由表进行操作 计算机间的数据通信是通过网络来实现的&#xff0c;路由就是从源主机到目标主机的转发过程 路由分为静态路由与动态路由&#xff0c;linux中的均为静态路由&#xff0c;动态路由由交换机路由器自动分配规则而来

[word] word悬挂缩进怎么设置? #经验分享#职场发展#经验分享

word悬挂缩进怎么设置&#xff1f; 在编辑Word的时候上方会有个Word标尺&#xff0c;相信很多伙伴都没使用过。其实它隐藏着很多好用的功能&#xff0c;今天就给大家分享下利用这个word标尺的悬挂缩进怎么设置&#xff0c;一起来看看吧&#xff01; 1、悬挂缩进 选中全文&…

Linux Mint 默认禁用未经验证的 Flatpak 软件包

Linux Mint 默认禁用未经验证的 Flatpak 软件包 Linux Mint 新政策 Linux Mint 项目宣布了一项新政策&#xff0c;即默认禁用那些未经官方验证的 Flatpak 软件包&#xff0c;以增强用户的安全保障。 当用户选择启用未经验证的 Flatpak 软件包时&#xff0c;Linux Mint 的软…

JAVA开发的一套(智造制造领航者云MES系统成品源码)saas云MES制造执行系统源码,全套源码,支持二次开发

JAVA开发的一套&#xff08;智造制造领航者云MES系统成品源码&#xff09;saas云MES制造执行系统源码&#xff0c;全套源码&#xff0c;支持二次开发 1990年11月&#xff0c;美国先进制造研究中心AMR&#xff08;Advanced Manufacturing Research&#xff09;就提出了MES&#…

Linux守护进程揭秘-无声无息运行在后台

在Linux系统中&#xff0c;有一些特殊的进程悄无声息地运行在后台&#xff0c;如同坚实的基石支撑着整个系统的运转。它们就是众所周知的守护进程(Daemon)。本文将为你揭开守护进程的神秘面纱&#xff0c;探讨它们的本质特征、创建过程&#xff0c;以及如何重定向它们的输入输出…

国产主流软硬件厂商生态分析

国产领域主流厂商汇总 信创&#xff0c;即信息技术应用创新&#xff0c;由“信息技术应用创新工作委员会”于2016年3月4日发起&#xff0c;是专注于软硬件关键技术研发、应用与服务的非营利性组织。作为科技自强的关键力量&#xff0c;信创在我国信息化建设中占据核心地位&…

外部mysql导入

利用这个命令&#xff1a; mysql -u username -p database_name < file.sql 然后就这样。成功导入。

定个小目标之每天刷LeetCode热题(12)

这是一道简单题&#xff0c;使用位运算中的异或运算即可&#xff0c;异或运算有以下性质&#xff1a; 1、任何数异或 0 结果仍然是原来的数&#xff0c;即 a⊕0a 2、任何数和其自身做异或运算&#xff0c;结果是 0 所以我们只需要让数组里的所有元素进行异或运算得到的结果就…

探索风电机组:关键软件工具全解析

探索风电机组&#xff1a;关键软件工具全解析 随着可再生能源市场的迅猛发展&#xff0c;风电作为一种重要的可再生能源&#xff0c;其相关技术和工具也越来越受到重视。风电机组的设计、仿真、优化及运维等方面&#xff0c;都需要依靠一系列专业软件工具来实现。这些软件涵盖…

Netty中的ByteBuf使用介绍

ByteBuf有三类&#xff1a; 堆缓存区&#xff1a;JVM堆内存分配直接缓冲区&#xff1a;有计算机内存分配&#xff0c;JVM只是保留分配内存的地址信息&#xff0c;相对于堆内存方式较为昂贵&#xff1b;复合缓冲区&#xff1a;复合缓冲区CompositeByteBuf&#xff0c;它为多个B…

VS2019创建c++动态链接库dll与调用方法

VS2019创建c动态链接库dll与调用方法 1.点击文件-》新建-》项目&#xff0c;输入dll,选择具有导出项的(DLL)动态链接库 2.输入一个文件名&#xff1a;dll2 头文件.h 3.添加加减法函数&#xff1a; // 下列 ifdef 块是创建使从 DLL 导出更简单的 // 宏的标准方法。此 DLL 中的…

题解web

1.[LitCTF 2023]Follow me and hack me 1&#xff09;进入题目环境&#xff0c;提示get传参&#xff0c;post传参 2&#xff09;看看源码&#xff0c;也没啥 3&#xff09;直接用hackbar&#xff0c;传入对应参数即可得到FLAG 3&#xff09;但是扫描出来它后端还有东西&#x…

Vue 学习笔记 总结

Vue.js 教程 | 菜鸟教程 (runoob.com) 放一下课上的内容 Vue练习 1、练习要求和实验2的用户注册一样&#xff0c;当用户输入后&#xff0c;能在下方显示用户输入的各项内容&#xff08;不需要实现【重置】按钮&#xff09; 2、实验报告中的实验小结部分来谈谈用JS、jQuery和…

618数码产品有什么推荐?四大2024“宝藏”数码产品推荐!

随着618购物节的热情逐渐升温&#xff0c;你是否在繁多且诱人的商品海洋中迷失方向&#xff0c;难以找到那最心仪的宝贝&#xff1f;团团在此特别为你精心挑选了一系列经过亲身体验的优质好物。这些推荐不仅时尚前沿&#xff0c;更贴合你的日常生活需求&#xff0c;确保实用与品…

霸气的短视频:成都科成博通文化传媒公司

霸气的短视频&#xff1a;瞬间的力量与魅力 在数字化浪潮中&#xff0c;短视频以其独特的魅力迅速崛起&#xff0c;成为社交媒体的新宠。而在众多短视频中&#xff0c;那些充满霸气、让人热血沸腾的作品&#xff0c;总能引起广泛的关注和讨论。成都科成博通文化传媒公司将从内…

CSS基础知识汇总

目录 CSS 基础知识1. CSS 的基本结构2. 选择器3. 常用 CSS 属性4. CSS 单位5. CSS 盒模型 总结 学习 CSS&#xff08;Cascading Style Sheets&#xff09;是前端开发的重要部分&#xff0c;它用于控制网页的样式和布局。以下是学习 CSS 过程中需要掌握的基本概念、符号和对应的…

Vue08-数据代理

一、Object.defineProperty() Object.defineProperty() 是 JavaScript 中的一个方法&#xff0c;用于直接在一个对象上定义一个新属性&#xff0c;或者修改一个对象的现有属性&#xff0c;并返回这个对象。 这个方法允许你精确地控制一个对象的属性&#xff0c;包括它的值、是…

数据库存储过程和锁机制

存储过程 存储过程是事先经过编译并存储在数据库中的一段SQL语句的集合,调用存储过程可以简化应用开发人员的很多工作,减少数据在数据库和应用服务器之间的传输,对于提高数据处理的效率是有好处的&#xff0c;存储过程思想上很简单,就是数据库SQL语言层面的代码封装与有重用 …

Flutter开发效率提升1000%,Flutter Quick教程之对Widget进行删除,剪切,粘贴

一&#xff0c;删除操作 1&#xff0c;首先我们选中要删除的Widget。 2&#xff0c;在左边的侧边栏&#xff0c;点击删除按钮&#xff0c;即可完成对组件的删除操作。 二&#xff0c;剪切。剪切是相同的道理&#xff0c;都是先选中&#xff0c;再点击对应的按钮。 1&#xff…