当前位置: 首页 > article >正文

Hutool TreeUtil 教程

一、简介Hutool 的TreeUtil是一个树结构工具类用于处理节点数据构建树形结构如菜单树、组织树、地区树等支持无限级分类。二、核心概念2.1 关键类TreeNode: 树节点配置类定义节点属性id、parentId、children等TreeUtil: 工具类提供构建树的方法三、快速开始3.1 添加依赖dependency groupIdcn.hutool/groupId artifactIdhutool-all/artifactId version5.8.26/version /dependency3.2 定义节点类import lombok.Data; import java.util.List; Data public class Menu { private String id; private String parentId; private String name; private Integer sort; private ListMenu children; }四、基本用法4.1 方式一使用 TreeNodeConfigimport cn.hutool.core.collection.CollUtil; import cn.hutool.core.lang.tree.Tree; import cn.hutool.core.lang.tree.TreeNodeConfig; import cn.hutool.core.lang.tree.TreeUtil; public class TreeUtilDemo { public static void main(String[] args) { // 构建节点列表 ListMenu menus buildMenuList(); // 配置树节点 TreeNodeConfig config new TreeNodeConfig(); config.setIdKey(id); // 节点ID字段名 config.setParentIdKey(parentId); // 父节点ID字段名 config.setWeightKey(sort); // 排序字段 config.setChildrenKey(children); // 子节点字段名 config.setDeep(3); // 最大深度 // 构建树 ListTreeString treeList TreeUtil.build(menus, 0, config, (menu, tree) - { tree.setId(menu.getId()); tree.setParentId(menu.getParentId()); tree.setName(menu.getName()); tree.setWeight(menu.getSort()); // 添加额外属性 tree.putExtra(extra, 自定义值); }); System.out.println(JSONUtil.toJsonPrettyStr(treeList)); } }4.2 方式二简化版// 如果节点类实现了 TreeNode 接口 ListTreeString treeList TreeUtil.build(menus, 0);五、完整示例5.1 构建菜单树import cn.hutool.core.lang.tree.Tree; import cn.hutool.core.lang.tree.TreeNodeConfig; import cn.hutool.core.lang.tree.TreeUtil; import cn.hutool.json.JSONUtil; import lombok.Data; import java.util.ArrayList; import java.util.List; Data public class Menu { private String id; private String parentId; private String name; private Integer sort; private String path; } public class MenuTreeBuilder { public static ListTreeString buildMenuTree(ListMenu menus) { // 配置 TreeNodeConfig config new TreeNodeConfig(); config.setIdKey(id); config.setParentIdKey(parentId); config.setWeightKey(sort); // 构建树 return TreeUtil.build(menus, 0, config, (menu, tree) - { tree.setId(menu.getId()); tree.setParentId(menu.getParentId()); tree.setName(menu.getName()); tree.setWeight(menu.getSort()); // 自定义字段 tree.putExtra(path, menu.getPath()); tree.putExtra(type, menu); }); } public static void main(String[] args) { ListMenu menus new ArrayList(); // 一级菜单 Menu menu1 new Menu(); menu1.setId(1); menu1.setParentId(0); menu1.setName(系统管理); menu1.setSort(1); menu1.setPath(/system); // 二级菜单 Menu menu11 new Menu(); menu11.setId(11); menu11.setParentId(1); menu11.setName(用户管理); menu11.setSort(1); menu11.setPath(/system/user); Menu menu12 new Menu(); menu12.setId(12); menu12.setParentId(1); menu12.setName(角色管理); menu12.setSort(2); menu12.setPath(/system/role); menus.add(menu1); menus.add(menu11); menus.add(menu12); ListTreeString tree buildMenuTree(menus); System.out.println(JSONUtil.toJsonPrettyStr(tree)); } }5.2 输出结果[ { id: 1, parentId: 0, name: 系统管理, weight: 1, path: /system, type: menu, children: [ { id: 11, parentId: 1, name: 用户管理, weight: 1, path: /system/user, type: menu }, { id: 12, parentId: 1, name: 角色管理, weight: 2, path: /system/role, type: menu } ] } ]六、高级用法6.1 自定义根节点ID// 使用不同的根节点标识 ListTreeString tree TreeUtil.build(nodes, -1, config, (node, tree) - { // 映射逻辑 });6.2 带条件的树构建// 过滤无效节点 ListMenu validMenus menus.stream() .filter(m - m.getStatus() 1) // 只构建启用的菜单 .collect(Collectors.toList()); ListTreeString tree TreeUtil.build(validMenus, 0, config, ...);6.3 多根节点树// 自动识别多个根节点parentId 不在列表中的节点 ListTreeString tree TreeUtil.build(menus, null, config, ...);6.4 获取树中所有节点IDimport cn.hutool.core.lang.tree.Tree; import java.util.ArrayList; import java.util.List; public ListString getAllNodeIds(ListTreeString trees) { ListString ids new ArrayList(); for (TreeString tree : trees) { ids.add(tree.getId()); if (tree.getChildren() ! null) { ids.addAll(getAllNodeIds(tree.getChildren())); } } return ids; }七、常见问题7.1 节点顺序问题// 设置权重字段进行排序 config.setWeightKey(sort); // 或者使用 Comparator config.setComparator(Comparator.comparing(Tree::getWeight));7.2 循环引用问题确保数据本身没有循环引用如A的父ID是BB的父ID是ATreeUtil会处理这种情况但可能导致栈溢出。7.3 性能优化// 大量数据时使用并行流处理 ListTreeString tree TreeUtil.build(menus, 0, config, (menu, tree) - { // 映射逻辑 }); // TreeUtil 内部已优化无需额外处理八、实际应用场景8.1 地区选择器public ListTreeString buildRegionTree(ListRegion regions) { TreeNodeConfig config new TreeNodeConfig(); config.setIdKey(code); config.setParentIdKey(parentCode); config.setWeightKey(orderNum); return TreeUtil.build(regions, 0, config, (region, tree) - { tree.setId(region.getCode()); tree.setParentId(region.getParentCode()); tree.setName(region.getName()); tree.putExtra(level, region.getLevel()); tree.putExtra(zipCode, region.getZipCode()); }); }8.2 部门树public ListTreeString buildDeptTree(ListDepartment depts) { TreeNodeConfig config new TreeNodeConfig(); config.setIdKey(deptId); config.setParentIdKey(parentId); config.setWeightKey(orderNum); config.setDeep(5); return TreeUtil.build(depts, 0, config, (dept, tree) - { tree.setId(dept.getDeptId()); tree.setParentId(dept.getParentId()); tree.setName(dept.getDeptName()); tree.putExtra(leader, dept.getLeader()); tree.putExtra(phone, dept.getPhone()); tree.putExtra(status, dept.getStatus()); }); }九、注意事项数据类型统一: id和parentId建议使用相同数据类型如String或Long根节点标识: 确保根节点的parentId匹配指定的根ID值性能考虑: 大数据量10000时注意性能TreeUtil内部使用Map索引性能良好线程安全: TreeUtil本身是线程安全的但结果需要自行处理空值处理: 如果parentId为null会被当作根节点处理十、总结Hutool TreeUtil 的优势✅ 简单易用代码简洁✅ 性能优秀使用Map索引✅ 配置灵活支持自定义字段✅ 功能完善满足大部分树形结构需求

相关文章:

Hutool TreeUtil 教程

一、简介Hutool 的 TreeUtil 是一个树结构工具类,用于处理节点数据构建树形结构(如菜单树、组织树、地区树等),支持无限级分类。二、核心概念2.1 关键类TreeNode: 树节点配置类,定义节点属性(id、parentId、…...

Hutool ObjectUtil 教程

一、简介Hutool 的 ObjectUtil 是一个对象操作工具类&#xff0c;提供了一系列实用的对象处理方法&#xff0c;包括判空、比较、默认值、序列化等。二、Maven依赖<dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId>&…...

如何突破Intel CPU性能瓶颈:智能电压调节工具的终极指南

如何突破Intel CPU性能瓶颈&#xff1a;智能电压调节工具的终极指南 【免费下载链接】Universal-x86-Tuning-Utility Unlock the full potential of your Intel/AMD based device. 项目地址: https://gitcode.com/gh_mirrors/un/Universal-x86-Tuning-Utility 你是否曾被…...

Cesium地图服务商大比拼:在Vue3项目中如何选择并接入ArcGIS、Bing、OSM和国内天地图?

Vue3Cesium地图服务选型实战&#xff1a;从ArcGIS到天地图的深度对比与集成指南 在智慧城市、物流追踪和地理信息可视化领域&#xff0c;地图底图的选择直接影响着用户体验和系统性能。作为前端工程师&#xff0c;我们常常陷入这样的困境&#xff1a;ArcGIS的影像精度令人心动但…...

分钟搞懂深度学习AI:实操篇:ResNet

从 UI 工程师到 AI 应用架构者 13 年前&#xff0c;我的工作是让按钮在 IE6 上对齐&#xff1b; 13 年后&#xff0c;我用 fetch-event-source 订阅大模型的“思维流”&#xff0c;用 OCR 解锁图片中的文字——前端&#xff0c;正在成为 AI 产品的第一道体验防线。 最近&#x…...

数据结构--二叉树知识讲解

一、树 1.**树的概念与结构 ** 树是一种非线性的数据结构&#xff0c;它是由 n(n ≥ 0) 个有限结点组成的、具有层次关系的集合。 当 n 0 时&#xff0c;称为空树。当 n > 0 时&#xff0c;有且仅有一个特殊结点&#xff0c;称为根结点Root。除根结点外&#xff0c;其余…...

别再死记硬背!用‘看图说话’六步法搞定开关电源环路补偿(附波特图分析)

开关电源环路补偿实战&#xff1a;六步图形化设计法 电源工程师们是否曾对环路补偿设计感到无从下手&#xff1f;面对密密麻麻的公式推导和抽象的理论分析&#xff0c;很多从业者往往陷入"知其然而不知其所以然"的困境。本文将颠覆传统学习路径&#xff0c;通过独创的…...

Ollama+AnythingLLM构建本地知识库问答+OpenAPI调用

机器配置&#xff1a;处理器&#xff1a;13th Gen Intel(R) Core(TM) i5-13500H(2.60 GHz) 机带 RAM&#xff1a;32.0 GB (31.7 GB 可用) 系统类型&#xff1a;64 位操作系统, 基于 x64 的处理器一、构建本地问答知识库1、下载Ollamahttps://ollama.com/download安装完成打开cm…...

【DeepSeek】BL2加载BL3x

下面是详细的流程解析&#xff1a; 1. BL2 阶段&#xff08;可信启动加载器&#xff09; 职责&#xff1a;BL2 运行在 Trusted SRAM 中&#xff0c;主要负责加载后续阶段的镜像。动作&#xff1a; BL2 从存储设备&#xff08;如 Flash&#xff09;中读取 BL31&#xff08;EL3 R…...

DriveDreamer-Policy:一种统一生成与规划的几何-落地世界-行动模型

26年4月来自极佳科技、多伦多大学和香港中文大学的论文“DriveDreamer-Policy: A Geometry-Grounded World–Action Model for Unified Generation and Planning”。 近年来&#xff0c;世界-动作模型&#xff08;WAM&#xff09;应运而生&#xff0c;旨在连接视觉-语言-动作&a…...

CustomTkinter:解决Python GUI现代化渲染与跨平台适配的技术架构

CustomTkinter&#xff1a;解决Python GUI现代化渲染与跨平台适配的技术架构 【免费下载链接】CustomTkinter A modern and customizable python UI-library based on Tkinter 项目地址: https://gitcode.com/gh_mirrors/cu/CustomTkinter Python的Tkinter框架在桌面GUI…...

2025最权威的十大AI论文方案推荐榜单

Ai论文网站排名&#xff08;开题报告、文献综述、降aigc率、降重综合对比&#xff09; TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 要是针对维普检测系统的 AI 降重需求&#xff0c;那就得从文本特征调整这方面着手。首先呢&a…...

Python 7 天入门 day_05:示例代码跟着敲

本文介绍了Python常用内置函数(zip/map/abs/ord/hex/bin/pow/eval等)的应用场景&#xff0c;包括数据打包、类型转换、数学运算等。 通过示例讲解了自定义函数的开发方法&#xff0c;包括参数处理(*args/**kwargs)、递归调用和变量作用域。 最后演示了冒泡排序和快速排序两种经…...

mysql如何配置审计日志输出_mysql audit_log_format设置

audit_log_format 设置成 STATEMENT 还是 JSON&#xff1f;MySQL 审计日志的 audit_log_format 只支持两个值&#xff1a;NEWLINE&#xff08;已弃用&#xff09;、JSON&#xff0c;没有 STATEMENT 选项。官方文档里写的 “STATEMENT” 是旧版 MySQL Enterprise Audit 插件的遗…...

nli-MiniLM2-L6-H768在教育行业落地:学生问答自动归类与知识点匹配案例

nli-MiniLM2-L6-H768在教育行业落地&#xff1a;学生问答自动归类与知识点匹配案例 1. 项目背景与价值 在教育场景中&#xff0c;学生每天会提出大量问题&#xff0c;这些问题分散在不同平台、不同课程中。传统的人工分类方式效率低下&#xff0c;且难以实现知识点精准匹配。…...

算法训练营第七天 | 环形链表 扭捏快指针步步退,霸道慢指针狠狠追

今日算法题&#xff1a;142. 环形链表 II 编写代码前想法&#xff1a; 在刚看到题目的时候&#xff0c;我觉得题目重点是如何判断链表是否有环&#xff0c;我初步判断应该是利用while() 进行判断&#xff0c;但如果没有环&#xff0c;该利用什么条件来进行判断的退出&#xff0…...

前端开发者构建AI应用实战指南

1. 前端开发者如何构建AI应用&#xff1a;从入门到实战作为一名长期奋战在前端领域的开发者&#xff0c;我清晰地记得第一次尝试将AI能力整合进Web应用时的迷茫。面对TensorFlow.js的文档、各种API接口和模型部署选项&#xff0c;那种既兴奋又无从下手的感觉至今难忘。经过两年…...

UE5Varest发送https请求发不出去,收不到任何回复

不要勾选&#xff0c;设置好后必须重启才能生效...

如何快速提升网盘下载速度:8大平台完整解决方案

如何快速提升网盘下载速度&#xff1a;8大平台完整解决方案 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 &#xff0c;支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼云盘 /…...

c#如何使用Record类型_c#Record类型从入门到精通教程

Record 是带语义的不可变数据容器&#xff0c;启用值相等、init-only 属性、非空保障及自动生成 ToString/Equals/GetHashCode&#xff1b;误当普通 class 用易踩坑。Record 类型不是语法糖&#xff0c;是带语义的不可变数据容器Record 类型在 C# 9 中不是“更简洁的 class 写法…...

告别Excel配置表:在Unity中搭建Luban+Jenkins的自动化配置管线

Unity游戏开发&#xff1a;基于LubanJenkins的自动化配置管理实践 在游戏研发领域&#xff0c;配置管理一直是连接策划与程序的重要桥梁。传统Excel配置表工作流中&#xff0c;策划修改表格后需要手动通知程序重新导入&#xff0c;版本控制混乱&#xff0c;多人协作时冲突频发。…...

别再用错了!银河麒麟V10 SP2中Crontab的5个高级用法与3个典型误区

别再用错了&#xff01;银河麒麟V10 SP2中Crontab的5个高级用法与3个典型误区 在银河麒麟V10 SP2的日常运维中&#xff0c;Crontab作为定时任务管理的核心工具&#xff0c;其重要性不言而喻。然而&#xff0c;许多中高级用户在使用过程中&#xff0c;往往陷入一些常见误区&…...

《JAVA面经实录》- 权限管理框面试题

《JAVA面经实录》- 权限管理框面试题Java权限管理框架面试题&#xff08;23道高频题&#xff09;本文严格按照指定题目顺序&#xff0c;整理每道题的面试标准回答补充要点&#xff0c;贴合后端面试实战场景&#xff0c;语言简洁、重点突出&#xff0c;可直接用于备考&#xff0…...

如何在 Firebase Storage 中批量获取所有媒体文件的下载链接

本文详解 2023 年 firebase sdk v9 中正确列出并批量获取 storage 中所有媒体文件&#xff08;如图片&#xff09;下载 url 的标准方法&#xff0c;涵盖完整代码示例、常见错误分析及生产环境注意事项。 本文详解 2023 年 firebase sdk v9 中正确列出并批量获取 storage 中…...

2026届毕业生推荐的AI辅助论文助手推荐榜单

Ai论文网站排名&#xff08;开题报告、文献综述、降aigc率、降重综合对比&#xff09; TOP1. 千笔AI TOP2. aipasspaper TOP3. 清北论文 TOP4. 豆包 TOP5. kimi TOP6. deepseek 由于学术研究对效率跟质量有着双重 demands&#xff0c;论文 AI 工具已然成了科研工作者的关…...

终极网盘直链下载助手:8大平台满速下载的完整指南

终极网盘直链下载助手&#xff1a;8大平台满速下载的完整指南 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 &#xff0c;支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼云盘…...

# 发散创新:用Go语言打造绿色计算的高效任务调度器在当今算力爆炸的时代

发散创新&#xff1a;用Go语言打造绿色计算的高效任务调度器 在当今算力爆炸的时代&#xff0c;绿色计算已从理念走向实践。它不仅关乎节能减排&#xff0c;更体现在如何以更低能耗完成更高效率的任务处理。本文将通过一个真实可运行的 Go 语言项目——GreenScheduler&#xff…...

RBAC 与安全策略:集群权限控制的正确姿势

文章目录 1. 认证与授权:两道门的本质区别 1.1 用户身份的三种类型 1.2 X.509 证书认证的工作原理 2. RBAC 授权模型:四个核心对象 2.1 Role 与 ClusterRole:作用域差异 2.2 RoleBinding 的一个反直觉特性 2.3 聚合 ClusterRole:可扩展的权限体系 3. ServiceAccount:权限泄…...

不会写Prompt、功能太单一?这款AI太懂我

试过不下二十款AI对话工具&#xff0c;要么功能单一只能回答基础问题&#xff0c;要么定制化门槛太高不会写Prompt根本用不好&#xff0c;要么价格贵得离谱长期用吃不消。直到最近挖到科学对话这款全能科研工具&#xff0c;用了一个多月&#xff0c;确实解决了我一直以来不少问…...

MedPeer科研工具最优搭配指南

我整理了MedPeer所有会员套餐的核心权益&#xff0c;结合不同科研身份的真实需求给大家梳理一遍&#xff0c;帮你快速找到最适合自己的高性价比选择。MedPeer会员分为综合全能型和垂直功能型两大类&#xff0c;共15种套餐&#xff0c;覆盖科研全流程&#xff0c;支持年卡/月卡&…...