java实现多级目录树(递归实现)

news2025/3/15 0:18:53

一.应用场景

有时候需要我们后台给前台传树结构的数据,要怎么查询? 怎么返回数据呢?

二.数据库表设计以及数据内容(以部门举例)

id            主键
parent_id     父级部门id
depart_name   部门名词
sort          部门排序

在这里插入图片描述

三.实体类

@Data
public class KunKun implements Serializable {
    private static final long serialVersionUID = 1L;

	@ApiModelProperty("部门id")
	private String id;
	
	@ApiModelProperty("父级部门id")
	private String parentId;
	
	@ApiModelProperty("机构/部门名称")
	private String departName;
	
	@ApiModelProperty("子级部门的集合")
	private List<KunKun> children;
}

四.代码实现

public void selectKunTree() {
	 //查询部门信息,获取全部部门信息
	List<KunKun> kunList = xxxxMapper.selectList() ;
	
	//组装成树形结构
	//找到所有的一级分类(也就是树的根节点)
	List<KunKun> ikuns = kunList.stream().filter(ikun ->
		ikun.getParentId() == 0
		).map((menu) -> {
			menu.setChildren(getChildrens(menu,kunList));
			return menu;
		}).sorted((menu1,menu2)-> {//如果你查询的时候就排好序了就不用写这里
			return menu1.getSort() - menu2.getSort();
		}).collect(Collectors.tolist());
		
	//返回树结构的list
	return ikuns;
 }


//递归方法
private List<KunKun> getChildrens(KunKun root,List<KunKun> all){
	List<KunKun> children = all.stream().filter(ikun -> { 
		return ikun.getParentId() == root.getId();
	}).map(ikun  -> { //找子部门
		ikun.setChildren(getChildrens(ikun ,all));
		return categoryEntity;
	}).sorted( (menu1 ,menu2)
		return menu1.getSort - menu2.getSort() 
	}).collect(Collectors .tolist());
	
	return children;
}

=========================

多个一级id场景

在这里插入图片描述

实现思路

获取所有的分类。
获取所有分类的id集合。使用stream()实现,stream使用教程
获取一级分类信息。同样使用stream()实现。
循环一级分类,在循环中将一级分类添加子分类,并且将一级分类加入返回的树结构中(备注:不加入返回的树结构中也行,直接返回步骤3的分类信息一样)。
重点是步骤4中将一级分类添加子分类,并且子分类在添加子子分类,子子分类再添加子子子分类··········等等,实现过程使用递归即可。
步骤5的实现过程:写一个递归方法,往当前节点添加子节点,首先获取当前节点的字节点集合,然后把这个集合放入到当前节点子节点属性中,接着再次调用当前递归的方法,把刚获取到的子节点当成新当前节点,获取新当前节点的新子节点,注意再次调用当前递归的方法,把刚获取到的子节点当成新当前节点之前首先判断新当前节点有没有子节点(判断方法:获取当前节点的字节点数组,根据数组的size()>0?判断是否有子节点),如果没有就不用递归。
总结:1-4是数据准备,5-6是实现递归(当前节点添加子节点的递归)。

代码展示

  1. 主要思路代码
@GetMapping("/list")
    public Result list() {
        //所有的分类
        List<Category> categoryList = categoryService.list();
        //所有分类id集合
        List<String> idList = categoryList.stream().map(Category::getId).collect(Collectors.toList());
        //返回的树分类结果
        List<Category> treeCategory = new ArrayList<>();
        //一级分类目录
        List<Category> categories = categoryList.stream().filter(category -> !idList.contains(category.getParentId())).collect(Collectors.toList());
        //循环当前一级分类目录
        for (Category category : categories) {
            //给当前分类节点 添加 子分类节点
            addChild(categoryList,category);
            //当前分类添加完子节点分类之后,添加到返回的树结构中
            treeCategory.add(category);
        }
        //把返回的树结构返回
        return Result.success(categories);
    }

    /**
     * 给当前分类节点 添加 子分类节点
     * @param categoryList 所有的分类
     * @param category 当前分类节点
     */
    public void addChild( List<Category> categoryList,Category category){
        //循环所有的分类,获取当前节点的所有子节点
        List<Category> categoryListChild = categoryList.stream().filter(category1 -> category1.getParentId().equals(category.getId())).collect(Collectors.toList());
        //把当前分类的子节点添加到当前分类
        category.setChildList(categoryListChild);
        //再次调用本方法,把当前节点的子节点当成当前节点,继续添加子节点,备注:这样会造成一直循环
        categoryListChild.forEach(category1 -> {
            //添加一步,判断当前节点是否有子节点,没有就不循环递归
            if (haveChild(categoryList,category1)){
            	addChild(categoryList,category1);
 			}
        });
    }

    /**
     * 判断当前节点 是否存在 子节点
     * @param categoryList 所有的分类
     * @param category 当前节点
     */
    public boolean haveChild( List<Category> categoryList,Category category){
        //获取当前节点的子节点
        List<Category> categoryListChild = categoryList.stream().filter(category1 -> category1.getParentId().equals(category.getId())).collect(Collectors.toList());
        //子节点大于0则存在,否则不存在
        return categoryListChild==null&&categoryListChild.size()>0;

  1. 实体类代码展示
    备注:实体类代码中一定要有一个子节点数组。实体类对用最上面的实体类图片
@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
public class Category implements Serializable {

    private static final long serialVersionUID = 1L;
    @TableId(value = "id", type = IdType.ID_WORKER_STR)
    private String id;

    /**
     * 类别名称
     */
    private String name;

    /**
     * 上级id
     */
    private String parentId;

    /**
     * 分类级别
     */
    private Integer type;
    /**
     * 子节点数组
     */
    @TableField(exist = false)
    private List<Category> childList;
}

##吧数据处理返回:

{
    "code": 200,
    "msg": "操作成功",
    "data": [
        {
            "id": "1",
            "name": "电子产品",
            "parentId": "0",
            "type": 1,
            "childList": [
                {
                    "id": "5",
                    "name": "笔记本电脑",
                    "parentId": "1",
                    "type": 2,
                    "childList": [
                        {
                            "id": "21",
                            "name": "联想笔记本",
                            "parentId": "5",
                            "type": 3,
                            "childList": []
                        },
                        {
                            "id": "22",
                            "name": "外星人笔记本",
                            "parentId": "5",
                            "type": 3,
                            "childList": []
                        },
                        {
                            "id": "23",
                            "name": "戴尔笔记本",
                            "parentId": "5",
                            "type": 3,
                            "childList": []
                        }
                    ]
                },
                {
                    "id": "6",
                    "name": "手机",
                    "parentId": "1",
                    "type": 2,
                    "childList": [
                        {
                            "id": "24",
                            "name": "苹果手机",
                            "parentId": "6",
                            "type": 3,
                            "childList": []
                        },
                        {
                            "id": "25",
                            "name": "菠萝手机",
                            "parentId": "6",
                            "type": 3,
                            "childList": []
                        }
                    ]
                },
                {
                    "id": "7",
                    "name": "耳机",
                    "parentId": "1",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "8",
                    "name": "电子烟",
                    "parentId": "1",
                    "type": 2,
                    "childList": []
                }
            ]
        },
        {
            "id": "2",
            "name": "生活用品",
            "parentId": "0",
            "type": 1,
            "childList": [
                {
                    "id": "10",
                    "name": "椅子",
                    "parentId": "2",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "11",
                    "name": "床",
                    "parentId": "2",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "19",
                    "name": "牙膏",
                    "parentId": "2",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "20",
                    "name": "牙刷",
                    "parentId": "2",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "9",
                    "name": "桌子",
                    "parentId": "2",
                    "type": 2,
                    "childList": []
                }
            ]
        },
        {
            "id": "3",
            "name": "卫生用品",
            "parentId": "0",
            "type": 1,
            "childList": [
                {
                    "id": "12",
                    "name": "卫生纸",
                    "parentId": "3",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "13",
                    "name": "湿巾",
                    "parentId": "3",
                    "type": 2,
                    "childList": []
                }
            ]
        },
        {
            "id": "4",
            "name": "学习用品",
            "parentId": "0",
            "type": 1,
            "childList": [
                {
                    "id": "14",
                    "name": "电子书",
                    "parentId": "4",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "15",
                    "name": "听力光盘",
                    "parentId": "4",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "16",
                    "name": "实体书",
                    "parentId": "4",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "17",
                    "name": "钢笔",
                    "parentId": "4",
                    "type": 2,
                    "childList": []
                },
                {
                    "id": "18",
                    "name": "笔记本子",
                    "parentId": "4",
                    "type": 2,
                    "childList": []
                }
            ]
        }
    ]
}

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

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

相关文章

SQL28 计算用户8月每天的练题数量(date_format函数的用法)

代码 select day(date) as day ,count(question_id) as question_cnt from question_practice_detail where date_format(date,%Y-%m)2021-08 group by day知识点 day函数取出日期格式数值里的日期,month,year函数也是类似的作用date_format规定日期/时间的输出格式 %Y 年&am…

JS如何判断普通函数与异步(async)函数

这里可以先打印一下普通函数和异步&#xff08;async&#xff09;函数的结构&#xff0c;如下图 可以看出两者原型链&#xff0c;普通函数的原型链指向的是一个函数&#xff0c;异步&#xff08;async&#xff09;函数原型链指向的是一个AsyncFunction&#xff0c;这时就会想到…

java 培训班预定管理系统Myeclipse开发mysql数据库web结构jsp编程servlet计算机网页项目

一、源码特点 java 培训班预定管理系统是一套完善的java web信息管理系统 采用serlvetdaobean&#xff0c;对理解JSP java编程开发语言有帮助&#xff0c;系统具有完整的源代码和数据库&#xff0c;系统主要采用B/S模式开发。开发环境为TOMCAT7.0,Myeclipse8.5开发&#xf…

MATLAB导出图程序

本文将以代码的形式快速介绍MATLAB导出图到Paper 1 从simulation导出数 2 与simulation同源文件夹下创建导图m文件 代码如下&#xff1a; % 实验后的数据处理用 M-文件 % clear all % 清空工作空间 % close all      % 关闭所有图形窗口 % load adp.mat …

LeetCode刷题计划

LeetCode刷题计划 推荐 代码随想录&#xff1a;https://github.com/youngyangyang04/leetcode-master 卡码网 练习ACM模式 https://kamacoder.com/ 01 #include <iostream> using namespace std;int main() {int a ,b;while(cin>>a>>b){cout<<ab<…

15-k8s控制器资源-deployment/部署控制器

一、deployment部署控制器概念 在学习rc和rs控制器资源时&#xff0c;我们指导&#xff0c;这两个资源都是控制pod的副本数量的&#xff0c;但是&#xff0c;他们两个有个缺点&#xff0c;就是在部署新版本pod或者回滚代码的时候&#xff0c;需要先apply资源清单&#xff0c;然…

四川古力未来科技公司抖音小店:靠谱的新电商之旅

随着互联网的飞速发展&#xff0c;电商行业日新月异&#xff0c;新兴平台如抖音小店正成为消费者新的购物天堂。在众多抖音小店中&#xff0c;四川古力未来科技公司的店铺以其独特的魅力吸引了众多消费者的目光。那么&#xff0c;四川古力未来科技公司抖音小店到底靠不靠谱呢&a…

OpenCV识别人脸案例实战

使用级联函数 基本流程 函数介绍 在OpenCV中&#xff0c;人脸检测使用的是cv2.CascadeClassifier.detectMultiScale()函数&#xff0c;它可以检测出图片中所有的人脸。该函数由分类器对象调用&#xff0c;其语法格式为&#xff1a; objects cv2.CascadeClassifier.detectMul…

【GPT-2】论文解读:Language Models are Unsupervised Multitask Learners

文章目录 介绍zero-shot learning 零样本学习 方法数据Input Representation 结果 论文&#xff1a;Language Models are Unsupervised Multitask Learners 作者&#xff1a;Alec Radford, Jeff Wu, Rewon Child, D. Luan, Dario Amodei, I. Sutskever 时间&#xff1a;2019 介…

Open CASCADE学习|直纹曲面(ruled surface)

直纹曲面是一类特殊的曲面&#xff0c;在几何学和微分几何中都有研究。它的主要特性是&#xff0c;曲面上的每一点都有至少一条直线经过。换句话说&#xff0c;直纹曲面可以由一条直线通过连续运动构成。在三维欧几里德空间中&#xff0c;最常见的直纹曲面是平面、柱面和锥面&a…

专业140+总410+合工大合肥工业大学833信号分析与处理综合考研经验电子信息与通信工程,真题,大纲,参考书。

经过一年努力奋战&#xff0c;今年初试总分410&#xff0c;其中专业课833信号分析与处理综合&#xff08;ss和dsp&#xff09;140&#xff08;感谢信息通信Jenny老师去年的悉心指导&#xff09;&#xff0c;数一130&#xff0c;顺利上岸&#xff0c;被合工大录取&#xff0c;看…

第三节:基于 InternLM 和 LangChain 搭建你的知识库(课程笔记)

视频链接&#xff1a;https://www.bilibili.com/video/BV1sT4y1p71V/?vd_source3bbd0d74033e31cbca9ee35e111ed3d1 文档地址&#xff1a; https://github.com/InternLM/tutorial/tree/main/langchain 课程笔记&#xff1a; 1.仅仅包含训练时间点之前的数据&#xff0c;无法…

TOP100 图论

1.200. 岛屿数量 给你一个由 1&#xff08;陆地&#xff09;和 0&#xff08;水&#xff09;组成的的二维网格&#xff0c;请你计算网格中岛屿的数量。 岛屿总是被水包围&#xff0c;并且每座岛屿只能由水平方向和/或竖直方向上相邻的陆地连接形成。 此外&#xff0c;你可以…

第14讲投票帖子详情实现

投票帖子详情实现 后端,根据id查询投票帖子信息&#xff1a; /*** 根据id查询* param id* return*/ GetMapping("/{id}") public R findById(PathVariable(value "id")Integer id){Vote vote voteService.getById(id);WxUserInfo wxUserInfo wxUserInf…

Python如何实现定时发送qq消息

因为生活中老是忘记各种事情&#xff0c;刚好又在学python&#xff0c;便突发奇想通过python实现提醒任务的功能&#xff08;尽管TIM有定时功能&#xff09;&#xff0c;也可定时给好友、群、讨论组发送qq消息。其工作流程是&#xff1a;访问数据库提取最近计划——>根据数据…

Leetcode 剑指 Offer II 065.单词的压缩编码

题目难度: 中等 原题链接 今天继续更新 Leetcode 的剑指 Offer&#xff08;专项突击版&#xff09;系列, 大家在公众号 算法精选 里回复 剑指offer2 就能看到该系列当前连载的所有文章了, 记得关注哦~ 题目描述 单词数组 words 的 有效编码 由任意助记字符串 s 和下标数组 ind…

每日OJ题_二叉树dfs①_力扣2331. 计算布尔二叉树的值

目录 力扣2331. 计算布尔二叉树的值 解析代码 力扣2331. 计算布尔二叉树的值 2331. 计算布尔二叉树的值 难度 简单 给你一棵 完整二叉树 的根&#xff0c;这棵树有以下特征&#xff1a; 叶子节点 要么值为 0 要么值为 1 &#xff0c;其中 0 表示 False &#xff0c;1 表示…

IO流---缓冲流,转换流,打印流,序列化流

缓冲流 缓冲流&#xff08;Buffered Stream&#xff09;也被称为高效流&#xff0c;它是对基本的字节字符流进行增强的一种流。通过缓冲流&#xff0c;可以提高数据的读写能力。 在创建缓冲流对象时&#xff0c;会创建一个内置的默认大小的缓冲区数组。通过对缓冲区的读写&…

【Windows】删除 VHD 虚拟磁盘时提示“文件已在 System 中打开”的解决方法

一、原因 正如显示的那样&#xff0c;虚拟磁盘仍在被系统占用。因此我们需要断开磁盘与系统的连接。 二、解决方法 1. 在“开始”菜单中搜索“磁盘管理”&#xff0c;选择“创建并格式化硬盘分区”。 2. 右键点击需要删除的虚拟磁盘&#xff0c;选择“分离 VHD”。 3. 点击“…

OpenAI 发布文生视频大模型 Sora,AI 视频要变天了,视频创作重新洗牌!AGI 还远吗?

一、一觉醒来&#xff0c;AI 视频已变天 早上一觉醒来&#xff0c;群里和朋友圈又被刷屏了。 今年开年 AI 界最大的震撼事件&#xff1a;OpenAI 发布了他们的文生视频大模型 Sora。 OpenAI 文生视频大模型 Sora 的横空出世&#xff0c;预示着 AI 视频要变天了&#xff0c;视…