Java之遍历树状菜单

news2025/2/4 22:45:24

😇作者介绍:一个有梦想、有理想、有目标的,且渴望能够学有所成的追梦人。

🎆学习格言:不读书的人,思想就会停止。——狄德罗

⛪️个人主页:进入博主主页

🗼专栏系列:无

🌼欢迎小伙伴们访问到博主的文章内容,在浏览阅读过程发现需要纠正的地方,烦请指出,愿能与诸君一同成长!

目录

文章内容如下

✏️前言

✏️一、数据库表

✏️二、编写实体代码

✏️三、写一个 Mapper dao 接口

✏️四、写一个 MyBatis Mapper SQL语句

✏️五、写一个树状菜单类

✏️六、测试

✏️总结


文章内容如下


✏️前言

如果前端要实现动态菜单管理,实现动态的菜单权限管理,那么在后台实现菜单信息树状输出是必要,但问题是,我从数据库中查询出所有的菜单信息然后存放到‘list’集合中,一个没有经过任何‘处理’的集合,它存放进去的数据,只是存放进去了原始数据,比如:部门管理,那么部门管理的字菜单是财务部、市场部等,问题来了,这些数据都在同一张表,识别它们关联的仅仅只是一个xxID而已,在遍历的时候没有经过逻辑处理,输出的仅仅只是一堆原始数据信息。这个时候我们可以去写一个逻辑来处理,根据ID字段识别并构建父子菜单项之间的关系。这样,前端在遍历菜单信息时,就能够以一种直观且有序的方式展现各个菜单项,而不是简单地列出所有数据。通过这种结构化的菜单信息,前端可以更有效地实现菜单的动态更新和权限控制,为用户呈现清晰且符合权限的菜单视图。下面我们直接上案例


✏️一、数据库表

👉首先我们要有一张菜单表

需要注意的是:下面的菜单表仅供参考,博主我在写的时候,怀着快速且简洁明了的目的,快速创建的表,小伙伴在实际的项目中,一定要好好设计菜单表。这里的菜单表是为了后面树状菜单信息打印输出而写的。


-- 菜单表
CREATE TABLE menu_table
(
	menu_id int PRIMARY KEY AUTO_INCREMENT NOT NULL , -- 主键自增
	menu_order varchar(500) NULL , -- 层级ID
	menu_icon varchar(20) NULL , -- 菜单图标
	menu_name varchar(20) NULL , -- 菜单名称
	menu_parent varchar(20) NULL , -- 父菜单名称
	menu_url varchar(100) NULL , -- 菜单url
	menu_permissionmark varchar(50) NULL , -- 权限标识
	menu_addtime datetime DEFAULT current_timestamp  NULL , -- 菜单创建时间
	menu_level varchar(500) NULL , -- 菜单层级关联id(父级为0)
	menu_state int NULL DEFAULT 1  -- 菜单状态

)COMMENT '菜单表',ENGINE = INNODB DEFAULT CHARSET=utf8 ;	select * from menu_table

-- 添加数据
INSERT INTO menu_table(menu_order,menu_icon,menu_name,menu_parent,menu_url,menu_permissionmark,menu_level)
VALUES 
('1','layui-icon-heart','系统管理','父级','indexs1.html','process.manage.list',null),
('2','layui-icon-heart','安全设置管理','父级','indexs2.html','role.manage.list',null),
('3','layui-icon-heart','工作流程','父级','indexs1.html','process.manage.list',null),
('4','layui-icon-heart','人力资源管理','父级','indexs2.html','role.manage.list',null),
('5','layui-icon-heart','企业信息管理','父级','indexs1.html','process.manage.list',null),
('6','layui-icon-heart','通讯管理','父级','indexs2.html','role.manage.list',null),
('7','layui-icon-heart','工作管理','父级','indexs1.html','process.manage.list',null),
('8','layui-icon-heart','时间管理','父级','indexs2.html','role.manage.list',null),
('9','layui-icon-heart','办公管理','父级','indexs1.html','process.manage.list',null),
('10','layui-icon-heart','组织架构管理','父级','indexs2.html','role.manage.list',null),

('11','layui-icon-heart','用户管理','1子级','indexs1.html','process.manage.list','1'),
('12','layui-icon-heart','角色管理','1子级','indexs2.html','role.manage.list','1'),
('13','layui-icon-heart','权限管理','1子级','indexs1.html','process.manage.list','2'),
('14','layui-icon-heart','流程管理','1子级','indexs2.html','role.manage.list','3'),
('15','layui-icon-heart','考勤管理','1子级','indexs1.html','process.manage.list','4'),
('16','layui-icon-heart','公告管理','1子级','indexs2.html','role.manage.list','5'),
('17','layui-icon-heart','邮件管理','1子级','indexs1.html','process.manage.list','6'),
('18','layui-icon-heart','任务管理','1子级','indexs2.html','role.manage.list','7'),
('19','layui-icon-heart','日程管理','1子级','indexs1.html','process.manage.list','8'),
('20','layui-icon-heart','计划管理','1子级','indexs2.html','role.manage.list','9'),
('21','layui-icon-heart','文件管理','1子级','indexs1.html','process.manage.list','1'),
('22','layui-icon-heart','笔记管理','1子级','indexs2.html','role.manage.list','1'),
('23','layui-icon-heart','员工通讯管理','1子级','indexs1.html','process.manage.list','6'),
('24','layui-icon-heart','讨论管理','1子级','indexs2.html','role.manage.list','6'),
('25','layui-icon-heart','便签管理','1子级','indexs1.html','process.manage.list','1'),
('26','layui-icon-heart','部门管理','1子级','indexs2.html','role.manage.list','10'),
('27','layui-icon-heart','采购部','2子级','indexs2.html','role.manage.list','26'),
('28','layui-icon-heart','外出岗','3子级','indexs2.html','role.manage.list','27'),
('29','layui-icon-heart','出口进货岗','3子级','indexs2.html','role.manage.list','27'),
('30','layui-icon-heart','财务部','2子级','indexs2.html','role.manage.list','26'),
('31','layui-icon-heart','会计师岗位','3子级','indexs2.html','role.manage.list','30'),
('32','layui-icon-heart','收银员岗位','3子级','indexs2.html','role.manage.list','30'),
('33','layui-icon-heart','菜单管理','1子级','indexs2.html','role.manage.list','1'),
('34','layui-icon-heart','市场部','2子级','indexs2.html','role.manage.list','26'),
('35','layui-icon-heart','总经办','2子级','indexs2.html','role.manage.list','26')

这是数据库的一张菜单表


✏️二、编写实体代码

👉写一个 Java 菜单实体类

package nf.db.oa.oasystems.modle.menus.menu.entity;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.Date;
import java.util.List;

/**
 * @Author 半杯可可
 * @Date 2023/12/2023/12/4
 * @Description 菜单信息
 */
@Data
@NoArgsConstructor
@AllArgsConstructor
public class Menu {
    private Integer menuId ; // 主键自增
    private String menuOrder ; // 层级ID
    private String menuIcon ; // 菜单图标
    private String menuName ; // 菜单名称
    private String menuParent ; // 父菜单名称
    private String menuUrl ; // 菜单url
    private String menuPermissionMark ; // 权限标识
    private Date menuAddTime ; // 菜单创建时间
    private Integer menuState ; // 菜单状态
    private String menuLevel ; // 菜单层级关联id(父级为null)
    private List<Menu> menus ; 

}

✏️三、写一个 Mapper dao 接口

👉写一个 MyBatis 的 Mapper  dao 接口用于定义应用程序中的数据访问,确保应用程序与数据库之间的交互

   /***
     * 查询所有菜单数据
     * @return
     */
    List<Menu> listMenu() ;

✏️四、写一个 MyBatis Mapper SQL语句

👉在 Mapper 配置中的 select 标签实现SQL语句

    <resultMap id="menuMap" type="nf.db.oa.oasystems.modle.menus.menu.entity.Menu">
        <id property="menuId" column="menu_id"/>
        <result property="menuOrder" column="menu_order"/>
        <result property="menuIcon" column="menu_icon"/>
        <result property="menuName" column="menu_name"/>
        <result property="menuParent" column="menu_parent"/>
        <result property="menuUrl" column="menu_url"/>
        <result property="menuPermissionMark" column="menu_permissionmark"/>
        <result property="menuAddTime" column="menu_addtime" />
        <result property="menuLevel" column="menu_level" />
        <result property="menuState" column="menu_state" />
    </resultMap>
    <!-- 查询菜单 -->
    <select id="listMenu" resultMap="menuMap">
        SELECT
            menu_id,
            menu_order,
            menu_icon,
            menu_name,
            menu_parent,
            menu_url,
            menu_permissionmark,
            menu_addtime,
            menu_level,
            menu_state
        FROM menu_table;
    </select>

✏️五、写一个树状菜单类

👉这个树状菜单类就是这次讲解的‘核心’了,如何树状输出菜单信息,就靠这个类了

package nf.db.oa.oasystems.utils;
import nf.db.oa.oasystems.modle.menus.menu.entity.Menu;
import java.util.ArrayList;
import java.util.List;
/**
 * @Author 半杯可可
 * @Date 2023/12/2023/12/14
 * @Description Hello World
 *
 * 菜单树状工具
 */
public class DendriformMenuUtil {
    private List<Menu> menulist=new ArrayList<>();

    public DendriformMenuUtil(List<Menu> menulist){
        this.menulist=menulist;
    }

    /**
     * 建立树形结构
     * @return
     */
    public List<Menu> buildTree(){
        List<Menu> treeMenus = new ArrayList<Menu>();
        List<Menu> rootNodes = getRootNode();
        for (Menu rootNode : rootNodes) {
            if (rootNode.getMenuLevel()==null){
                rootNode.setMenuLevel("") ;
            }
            Menu menuNode = buildChildTree(rootNode);
            treeMenus.add(menuNode);
        }
        return treeMenus;
    }

    /**
     * 实现获取所有子目录
     * @param rootNode
     * @return
     */
    private Menu buildChildTree(Menu rootNode) {
        List<Menu> childMenus = new ArrayList<Menu>();
        for (Menu menuNode : menulist) {
            if (menuNode.getMenuLevel().equals(rootNode.getMenuOrder())) {
                childMenus.add(buildChildTree(menuNode));
            }
        }
        rootNode.setMenus(childMenus);
        return rootNode;
    }

    /**
     * 获取根节点
     * @return
     */
    private List<Menu> getRootNode() {
        List<Menu> rootMenuList = new ArrayList<Menu>();
        for (Menu menuNode : menulist) {
            if (menuNode.getMenuLevel()==null){
                menuNode.setMenuLevel("") ;
            }
            if (menuNode.getMenuLevel().isEmpty()) {
                rootMenuList.add(menuNode);
            }
        }
        return rootMenuList;
    }
}

上面我用于识别父子级的标识ID的数据类型是 String 字符串类型,你们要是 Integer 包装类或 int 类型,可以修改上面的一些代码,原理都一样的。


✏️六、测试

👉下面就是测试的代码了

   @Test
     public void listMenu(){
        List<Menu> list = dao.listMenu();
        DendriformMenuUtil dendriformMenuUtil = new DendriformMenuUtil(list) ;
        list=dendriformMenuUtil.buildTree();
        String jsonOutput = toJSONString(list);
        log.info("========>"+jsonOutput)
    }

测试结果图,这个结果图可以看出,这就是树状输出的格式了


✏️总结

树状菜单输出,其实说简单也不简单,说复杂也不复杂,实现这一个功能,有一个核心知识那就是‘递归’,想去了解的伙伴就去多了解、多学习,我们一起学习,一起成长!

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

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

相关文章

Android studio 使用greenDao根据实体类生成dao类

1.遇到的问题 使用android studio根据实体类生成dao其实也很简单&#xff0c;你只要实现 Parcelable Entity public class ConfigDataModel implements Parcelable {Id(autoincrement true)private Long id null; } 2.使用自带的方法生成 使用build-->make Project生成 …

软件体系结构复习

复习参考&#xff1a; 一文搞懂什么是RESTful API 九种常见UML图 企业级软件开发的基本概念 软件架构&#xff1a;在设计和构建软件系统时采用的基本结构和原则。 它涉及到对软件系统进行模块化、组织和分解的方式&#xff0c;以及不同模块之间的交互和通信方式。在软件设计中…

pyomo使用cplex求解,进行冲突校验

文章目录 求解参数设置模型保存模型冲突校验pyomo冲突校验cplex冲突校验docplex冲突校验 CPLEX 安装包下载 pyomo使用 cplex求解&#xff0c;进行冲突校验 求解参数设置 options {"timelimit" : 60*60, # 设置求解时间&#xff0c;超过设置时间&#xff0c;求解停…

短视频矩阵系统的崛起和影响

近年来&#xff0c;短视频矩阵系统已经成为了社交媒体中的一股新势力。这个新兴的社交媒体形式以其独特的魅力和吸引力&#xff0c;迅速吸引了大量的用户。这个系统简单来说就是将海量短视频整合在一个平台上&#xff0c;使用户可以方便地观看和分享好玩有趣的短视频。 短视频…

巨大成就背后,我揭开比亚迪销量第一秘密

可以说比亚迪已经成为我每日注视的焦点所在&#xff0c;无论数据走势还是旗下各品牌新车动向&#xff0c;我都抱着浓厚兴趣细细研究。 今天&#xff0c;当我读到比亚迪再次蝉联新能源车销量王座的消息时&#xff0c;我不禁为之振奋。这家在新能源车业风生水起的企业&#xf…

【Dart】P0 Win、Mac 使用与安装

Dart 使用与安装 Dart 下载安装Windows 版本MacOS版本处于境外安装 Dart 开发工具 Dart 下载安装 Windows 版本 安装网址&#xff1a; http://gekorm.com/dart-windows/ 安装后测试&#xff1a; dart --versionMacOS版本 首先安装 Homebrew&#xff1a; 终端输入&#xff…

【EI社科会议征稿】第四届公共管理与智能社会国际学术会议(PMIS 2024)

第四届公共管理与智能社会国际学术会议&#xff08;PMIS 2024) 2024 4th International Conference on Public Management and Intelligent Society 第四届公共管理与智能社会国际学术会议将在2024年3月15-17日在长沙召开。PMIS 2024由中南大学社会计算研究中心、中南大学公共…

线上夺旗开启!由MoveBit主办,Sui独家赞助的MoveCTF 2024开放注册

*本文从MoveBit公众号转载 去年底&#xff0c;由 MoveBit 推出的首个 MoveCTF 线上安全竞赛 — MoveCTF 2022 备受关注&#xff0c;吸引了大量 Move 生态开发者踊跃参与。为持续推动 Move 生态系统的蓬勃发展&#xff0c;进一步丰富开发者在 Move 安全的专业知识&#xff0c;M…

到底是前端验证还是后端验证

背景 软件应用研发中&#xff0c; 前端验证还是后端验证这是意识与认知问题。鉴于某些入门同学还不清楚&#xff0c;我们再来看下&#xff1a; 一. 从软件行业来自国外 Q: 前端验证和后端验证都是对同一个数据的验证&#xff0c;有什么区别&#xff1f; A: 二者的目的不同&…

Mongodb基础介绍与应用场景

NoSql 解决方案第二种 Mongodb MongoDB 是一款开源 高性能 无模式的文档型数据库 当然 它是NoSql数据库中的一种 是最像关系型数据库的 非关系型数据库 首先 最需要注意的是 无模式的文档型数据库 这个需要后面我们看到它的数据才能明白 其次是 最像关系型数据库的非关系型数据…

(企业 / 公司项目)微服务OpenFeign怎么实现服务间调用?(含面试题)

Feign: 远程调用组件使用步骤&#xff0c;理解上面的图  后台系统中, 微服务和微服务之间的调用可以通过Feign组件来完成.  Feign组件集成了Ribbon负载均衡策略(默认开启的, 使用轮询机制),Hystrix熔断器 (默认关闭的, 需要通过配置文件进行设置开启)  被调用的微服务…

Seem环境安装

创建虚拟环境 conda create -n seem python3.8 conda activate seem 安装相关依赖&#xff1a;&#xff08;不按照的话会报错&#xff09; sudo apt-get install openmpi-bin libopenmpi-devconda install gcc_linux-64pip install mpi4py 导入环境 export PYTHONPATH$(pwd…

为工业机器人设计提供解决方案——Samtec砷泰连接器

【摘要/前言】 机器人已经与人类社会相处了数十年&#xff0c;这里指的机器人Robots并不是那些登上头条或在技术大会上引起轰动的可爱或未来主义的设计。它们是在幕后工作的机器&#xff0c;负责制造我们日常生活中使用的许多产品。 【机器人设计遇到的难题】 按照今天的标准…

图片批量处理:图片批量缩放,高效调整尺寸的技巧

在数字媒体时代&#xff0c;图片处理已是日常生活和工作中不可或缺的一部分。有时候要批量处理图片&#xff0c;如缩放图片尺寸&#xff0c;以满足不同的应用需求。现在一起来看看办公提效式具如何高效的将图片批量处理方法&#xff0c;快速、准确地批量调整图片尺寸操作。 下…

饥荒Mod 开发(二四):制作一把万能工具

饥荒Mod 开发(二三)&#xff1a;显示物品栏详细信息 源码 饥荒中的每种工具都有独特的功能&#xff0c;比如 斧头用来砍树&#xff0c; 铲子用来 挖东西&#xff0c;鹤嘴锄用来挖矿&#xff0c; 锤子可以敲碎东西&#xff0c;所以我们随身备着4种工具&#xff0c;不仅需要多占用…

浅谈智能型电动机保护器在也门化工行业的应用

1.背景信息Background 现代化工工业中&#xff0c;电动机作为一种拖动机械&#xff0c;成为所有动力机械基础&#xff0c;科学技术不断进步和工艺控制不断完善&#xff0c;尤其是自动化生产要求&#xff0c;迫切需要开发和完善电动机控制和保护设备&#xff0c;实现对生产过程…

论文降重隐藏字符怎么识别 papergpt

大家好&#xff0c;今天来聊聊论文降重隐藏字符怎么识别&#xff0c;希望能给大家提供一点参考。 以下是针对论文重复率高的情况&#xff0c;提供一些修改建议和技巧&#xff0c;可以借助此类工具&#xff1a; 论文降重隐藏字符的识别方法 一、引言 在论文降重过程中&#xff…

Linux 学习

复制/etc 文件夹到/mnt 目录 cp -r(-a) /etc /mnt回到上一次文件夹 cd -切换到当前用户的家目录_cd ~________________________如何查找ls 命令的位置_______which ls_________________________________请写出ll 命令中查看到的7大文件类型缩写 - s l p c b …

PostgreSQL | FunctionProcedure | 函数与存储过程的区别

文章目录 PostgreSQL | Function&Procedure | 函数与存储过程的区别1. 简述书面说法大白话讲 2. 函数&#xff08;Function&#xff09;2.1 定义2.2 用途2.3 执行2.4 事务处理2.5 说点例子1. 当参数都是IN类时2. 参数中出现OUT、INOUT参数时 3. 存储过程&#xff08;Proced…