Java 读取Excel导入数据库,形成树状结构

最近开发过程中遇到一个Excel的导入的功能,因为导入的数据结构具有层次结构,经过一番研究,最终得以实现,所有写下该文章,记录过程,供以后参考。

下图是导入Excel的数据结构:

使用POI解析Excel,数据封装然后进行入库。下面是核心代码。

@Overridepublic KnowledgeBaseDictImportRespVO importKnowledgeBaseDict(MultipartFile file) throws IOException {KnowledgeBaseDictImportRespVO kbdir = KnowledgeBaseDictImportRespVO.builder().build(); // 该对象是方法返回的结果对象Map<String,String> res = new ConcurrentHashMap<>();List<String> createProblemsList = new ArrayList<>();String fileName = file.getOriginalFilename();String fileType = fileName.substring(fileName.lastIndexOf(".")).toLowerCase();if(fileType.equals(".xls") || fileType.equals(".xlsx")) {AdminUserRespDTO adminUserRespDTO = adminUserApi.getUser(SecurityFrameworkUtils.getLoginUserId()).getData();//poi解析excelInputStream inputStream = file.getInputStream();// 读取工作薄 .xls 与 .xlsx 需要创建不通的工作簿 采用工作簿工厂类创建对应的工作簿类Workbook workbook = WorkbookFactory.create(inputStream);Sheet sheet = workbook.getSheetAt(0);//获取系统名称Row row0 = sheet.getRow(0); // 第一行的表头Cell cell0 = row0.getCell(0); // 第一列String belongSystem = cell0.getStringCellValue().trim().replace("系统功能树","");if (StringUtils.isEmpty(belongSystem)) {res.put(StrUtils.getRandomString(6),"归属系统不能为空!");kbdir.setFailureProblems(res);return kbdir;}KnowledgeBaseDictDO kbdictdo = knowledgeBaseDictMapper.getKnowledgeBaseDictByParenetIdAndMenuName(0L,belongSystem);//定义一个cellId,此为每一次循环前一列的idlong cellId = 0;String cellName = null; // 记录父级名称String firstMenuName = null; // 因为合并单元格只有第一个有值,需要记录该值。int count = 0;if (ObjectUtils.isEmpty(kbdictdo)) {kbdictdo = new KnowledgeBaseDictDO();kbdictdo.setSort(0);kbdictdo.setParentId(0L);kbdictdo.setMenuLevel(0);kbdictdo.setMenuName(belongSystem);kbdictdo.setBelongingSystem(belongSystem);kbdictdo.setCreator(adminUserRespDTO.getNickname());int rs = knowledgeBaseDictMapper.insert(kbdictdo);if (rs > 0) {count += rs;createProblemsList.add(belongSystem);cellId = kbdictdo.getId();cellName = kbdictdo.getMenuName();}}int rows = sheet.getPhysicalNumberOfRows();if (rows > 2) {//按照行进行循环,读取当前行的列for (int i = 0,j = 2; j < rows; j++,i++) {// 读取行Row row = sheet.getRow(j);//查询当前行有多少列int physical = sheet.getRow(j).getPhysicalNumberOfCells();if (row != null) {//获取第一列的单元格Cell cell = row.getCell(0);//判断单元格是否为空if (!(cell == null || "".equals(cell.toString().trim()))) {firstMenuName = cell.getStringCellValue().trim();//根据单元格的数据查询数据库是否存在记录KnowledgeBaseDictDO kbddoFirst = knowledgeBaseDictMapper.getKnowledgeBaseDictByMenuNameAndMenuLevelAndbls(firstMenuName,1,belongSystem);//数据库不存在,添加记录if (ObjectUtils.isEmpty(kbddoFirst)) {kbddoFirst = new KnowledgeBaseDictDO();kbddoFirst.setSort(i);kbddoFirst.setParentId(cellId);kbddoFirst.setBelongingSystem(belongSystem);kbddoFirst.setParentMenuName(cellName);kbddoFirst.setMenuName(cell.getStringCellValue().trim());kbddoFirst.setMenuLevel(1);kbddoFirst.setCreator(adminUserRespDTO.getNickname());int rs = knowledgeBaseDictMapper.insert(kbddoFirst);if (rs > 0) {count += rs;createProblemsList.add(cell.getStringCellValue().trim());// 记录父级id 与 名称cellId = kbddoFirst.getId();cellName = kbddoFirst.getMenuName();}} else {//数据库存在记录,将这条记录的id作为父idcellId = kbddoFirst.getId();cellName = kbddoFirst.getMenuName();}} else { // 表示循环到合并单元格的地方,查询数据库记录父级信息if (StringUtils.isNotEmpty(firstMenuName)) {KnowledgeBaseDictDO kbddoFirst = knowledgeBaseDictMapper.getKnowledgeBaseDictByMenuNameAndMenuLevelAndbls(firstMenuName,1,belongSystem);cellId = kbddoFirst.getId();cellName = kbddoFirst.getMenuName();}}}// 从第二列单元格开始for (int k = 1,l = 0; k < physical; k++,l++) {//取单元格Cell cell = row.getCell(k);//判断单元格是否为空if (!(cell == null || "".equals(cell.toString().trim()))) {//查询数据库有无此记录KnowledgeBaseDictDO kbddoSecond = knowledgeBaseDictMapper.getKnowledgeBaseDictByMenuNameAndMenuLevelAndbls(cell.getStringCellValue().trim(),k + 1,belongSystem);if (ObjectUtils.isEmpty(kbddoSecond)) {kbddoSecond = new KnowledgeBaseDictDO();kbddoSecond.setSort(l);kbddoSecond.setParentId(cellId);kbddoSecond.setBelongingSystem(belongSystem);kbddoSecond.setParentMenuName(cellName);kbddoSecond.setMenuName(cell.getStringCellValue().trim());kbddoSecond.setMenuLevel(k+1);kbddoSecond.setCreator(adminUserRespDTO.getNickname());int rs = knowledgeBaseDictMapper.insert(kbddoSecond);if (rs > 0) {count += rs;createProblemsList.add(cell.getStringCellValue().trim());// 记录父级id 与 名称cellId = kbddoSecond.getId();cellName = kbddoSecond.getMenuName();}} else {cellId = kbddoSecond.getId();cellName = kbddoSecond.getMenuName();}if (k == physical - 1) {// 内层循环结束,表示一行的一级菜单结束,需要指定一级菜单的父级cellId = kbdictdo.getId();cellName = kbdictdo.getMenuName();}} else { // 如果为空,需要记录父级id与名称cellId = kbdictdo.getId();cellName = kbdictdo.getMenuName();}}}} else {res.put(StrUtils.getRandomString(6),"导入的数据不能为空!");kbdir.setFailureProblems(res);return kbdir;}kbdir.setCreateProblems(createProblemsList);kbdir.setFailureProblems(res);}return kbdir;}

 该方法也适用于下面的Excel数据结构导入(未进行列合并单元格的数据结构):

树形结构数据如何高效封装,通过接口返回前端展示,请看我的另一篇文章。

Java 树形结构数据如何高效返回给前端进行展示?-CSDN博客

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

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

相关文章

被拷打已老实!面试官问我 #{} 和 ${} 的区别是什么?

引言&#xff1a;在使用 MyBatis 进行数据库操作时&#xff0c;#{} 和 ${} 的区别是面试中常见的问题&#xff0c;对理解如何在 MyBatis 中安全有效地处理 SQL 语句至关重要。正确使用这两种占位符不仅影响应用的安全性&#xff0c;还涉及到性能优化。 题目 被拷打已老实&…

浅谈RC4

一、什么叫RC4&#xff1f;优点和缺点 RC4是对称密码&#xff08;加密解密使用同一个密钥&#xff09;算法中的流密码&#xff08;一个字节一个字节的进行加密&#xff09;加密算法。 优点&#xff1a;简单、灵活、作用范围广&#xff0c;速度快 缺点&#xff1a;安全性能较差&…

操作系统真象还原:输入输出系统

第10章-输入输出系统 这是一个网站有所有小节的代码实现&#xff0c;同时也包含了Bochs等文件 10.1 同步机制–锁 10.1.1 排查GP异常&#xff0c;理解原子操作 线程调度工作的核心内容就是线程的上下文保护&#xff0b;上下文恢复 。 根本原因是访问公共资源需要多个操作&…

超级数据查看器 教程pdf 1-31集 百度网盘

百度网盘链接 提取码1234https://pan.baidu.com/s/1s_2lbwZ2_Su83vDElv76ag?pwd1234 通过百度网盘分享的文件&#xff1a;超级数据查看器 … 链接:https://pan.baidu.com/s/1s_2lbwZ2_Su83vDElv76ag?pwd1234 提取码:1234 复制这段内容打开「百度网盘APP 即可获取」

Python接口测试实战之搭建自动化测试框架

&#x1f345; 视频学习&#xff1a;文末有免费的配套视频可观看 &#x1f345; 点击文末小卡片&#xff0c;免费获取软件测试全套资料&#xff0c;资料在手&#xff0c;涨薪更快 一.数据分离:从Excel中读取数据 之前的用例中&#xff0c;数据直接写在代码文件里&#xff0c;不…

Day28:回溯法 491.递增子序列 46.全排列 47.全排列 II 332.重新安排行程 51. N皇后 37. 解数独 蓝桥杯 与或异或

491. 非递减子序列 给你一个整数数组 nums &#xff0c;找出并返回所有该数组中不同的递增子序列&#xff0c;递增子序列中 至少有两个元素 。你可以按 任意顺序 返回答案。 数组中可能含有重复元素&#xff0c;如出现两个整数相等&#xff0c;也可以视作递增序列的一种特殊情…

13.1.资源清单的管理工具-helm

目录 一、helm的介绍 1.helm的价值概述 2.helm的关键名词 二、安装部署helm 1.解压安装包并设置全局命令 2.添加命令补全 三、使用helm部署服务管理 1.使用helm创建chart 1.1创建工作目录 1.2.helm创建chart 2.响应式创建名称空间 3.安装chart到名称空间 4.使用hel…

PHP转Go系列 | 字符串的使用姿势

大家好&#xff0c;我是码农先森。 输出 在 PHP 语言中的输出比较简单&#xff0c;直接使用 echo 就可以。此外&#xff0c;在 PHP 中还有一个格式化输出函数 sprintf 可以用占位符替换字符串。 <?phpecho 码农先森; echo sprintf(码农:%s, 先森);在 Go 语言中调用它的输…

科研——BIBM论文修改和提交

文章目录 引言投递流程Latex翻译流程latex模板使用bib文件正文修改 反馈时间线等待审稿结果 引言 第一轮投递快结束了&#xff0c;这里得加快进度&#xff0c;二十号截至&#xff0c;这里得在截至之前投一下&#xff0c;这里翻译整理一下投递的流程 投递流程 投递链接论文是…

js文件导出功能

效果图&#xff1a; 代码示例&#xff1a; <!DOCTYPE html> <html> <head lang"en"><meta charset"UTF-8"><title>html 表格导出道</title><script src"js/jquery-3.6.3.js"></script><st…

LoRA用于高效微调的基本原理

Using LoRA for efficient fine-tuning: Fundamental principles — ROCm Blogs (amd.com) 大型语言模型的低秩适配&#xff08;LoRA&#xff09;用于解决微调大型语言模型&#xff08;LLMs&#xff09;的挑战。GPT和Llama等拥有数十亿参数的模型&#xff0c;特定任务或领域的微…

【因果推断python】44_评估因果模型2

目录 累积弹性曲线 累积增益曲线 考虑差异 关键思想 累积弹性曲线 再次考虑将价格转换为二元处理的说明性示例。我们会从我们离开的地方拿走它&#xff0c;所以我们有弹性处理带。我们接下来可以做的是根据乐队的敏感程度对乐队进行排序。也就是说&#xff0c;我们把最敏感…

韩顺平0基础学java——第26天

p523-547 HashSet扩容时&#xff0c;只要节点到达了阈值就会扩&#xff0c;而不是数组长度到了才扩。 比如长16的数组&#xff0c;索引1放了8个&#xff0c;索引3放了4个&#xff0c;我再加一个他就会扩容。 另外谁能告诉我老师的debug界面是怎么设置的吗忘光了 HashSet存放…

基于Spring+Vue的前后端分离的计算器

麻雀虽小&#xff0c;五脏俱全 该项目已部署上线&#xff1a;http://calculator.wushf.top/ 并通过Gitee Go流水线实现持续部署。 需求分析 表达式求值 支持加减乘除四则运算、支持高精度 获取日志 Api文档定义 前后端分离&#xff0c;人不分离 通过Apifox定义接口细节&#…

(el-Transfer)操作(不使用 ts):Element-plus 中 Select 组件动态设置 options 值需求的解决过程

Ⅰ、Element-plus 提供的Select选择器组件与想要目标情况的对比&#xff1a; 1、Element-plus 提供Select组件情况&#xff1a; 其一、Element-ui 自提供的Select代码情况为(示例的代码)&#xff1a; // Element-plus 提供的组件代码: <template><div class"f…

闹大了!高考作文“人工智能与AI”引发争议,专家喊话,部分考生家长无奈,直呼:“太不公平了!这哪里是考作文,分明是在考城乡差距啊!”

闹大了&#xff01;高考作文“人工智能与AI”引发争议&#xff0c;专家喊话&#xff0c;部分考生家长无奈&#xff0c;直呼&#xff1a;“太不公平了&#xff01;这哪里是考作文&#xff0c;分明是在考城乡差距啊&#xff01;” ​高考&#xff0c;本该是最公平的战场&#xff…

IO流2.

字符流-->字符流的底层其实就是字节流 public class Stream {public static void main(String[] args) throws IOException {//1.创建对象并关联本地文件FileReader frnew FileReader("abc\\a.txt");//2.读取资源read()int ch;while((chfr.read())!-1){System.out…

MySQL系列-语法说明以及基本操作(一)

1、前言 主要讲解MySQL的基本语法 官网文档 https://docs.oracle.com/en-us/iaas/mysql-database/doc/getting-started.html 关于MySQL的基本语法&#xff0c;关于数据类型、表的操作、数据操作、事务、备份等&#xff0c;可参考 http://www.voidme.com/mysql 2、数据类型 数…

多环境镜像晋级/复用最佳实践

作者&#xff1a;木烟 本文主要介绍镜像构建部署场景&#xff0c;多环境镜像晋级/复用最佳实践&#xff0c;保证“所发即所测”。 场景介绍 应用研发场景有效地管理镜像产物是确保软件快速、安全、可靠部署的关键环节。通常一个应用研发需要经过测试、预发、生产各个阶段&am…

PHP转Go系列 | 变量常量的使用姿势

大家好&#xff0c;我是码农先森。 变量 在 PHP 语言中&#xff0c;初始化变量虽然只有一行&#xff0c;其实包含了两步&#xff0c;一是声明变量&#xff0c;二是赋值给变量&#xff0c;同一个变量可以任意再赋值任何类型的数据。 <?php// 初始化变量 $name "man…