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

EasyExcel模板导出与公式计算(下)

目录

环境要求

功能预览

需求分析

导入依赖

制作模板

编写代码

格式优化

最终效果

总结


在上一篇 EasyExcel模板导出与公式计算(上)-CSDN博客 文章中我们知道了在若依中使用自带的@Excel注解来实现表格数据的导出,并且通过重写相关接口来支持导出公式的自动计算功能,而在本文中,我们将探究如何在带有自定义表头的定制化场景下,如何实现表格数据的导出和公式的自动计算。

环境要求

若依Cloud 3.6.3

EasyExcel  3.3.2

功能预览

与上一篇文章相同,这里继续以若依用户管理的导出功能为示例,做一个简单的公式计算的验证,大家可根据自己需求修改相应代码。

首先本地启动若依微服务版本,登录后台,依次选择系统管理 —> 用户管理菜单,进入如下页面

 点击该页面的导出按钮

获取如下导出文件

现在如果我们想要在第四行计算第二和第三行用户编号的和,可以直接在B4单元格输入"=A2+A3"这样的公式来实现,可以注意到此时B4单元格中的公式颜色已经发生变化,并且自动选中了A2和A3单元格,但实际上我们的需求为从数据库表中导出的单元格实现自动计算结果,接下来将通过若依前端页面来实现该功能验证。

现在需求变为除了能够正确计算单元格数据外,还要变成像如下图的导出样式:

不难看出,我们需要导出的结果有自定义的表头,以及时间,并且需要特别注意的是,原本的计算公式由=A2+A3,变为了=A5+A6,这是因为新增了表头导致行号发生了变化,接下来将针对这种需求进行实现。

需求分析

由于有了自定义标头的要求,显然我们不能按照上一篇文章中直进行数据的导出,因此我们将使用EasyExcel实现该功能

导入依赖

首先在ruoyi-common-core的pom.xml添加EaysExcel的依赖,之所以添加到这里是因为该模块为通用核心模块,会在项目中被其他的模块引用,可以实现“一处添加,多处使用”的效果。

<!-- easyexcel --><dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>3.3.2</version></dependency>

制作模板

为了实现上述中的导出效果,我们需要使用Excel制作一个用于导出的模板,最终效果如下所示:

可以看到图中红色框所标识的是新增的表头内容,其中时间后面的{dataDate} 则为填充对象,而第5行对应的内容中除了花括号外,还有一个点,这其实是代表的是将使用对象集合填充,与时间的dataDate区别为, dataDate仅为一个对象。

编写代码

由于我们使用了EasyExcel之后,原有的导出功能需要做调整, @Excel注解将被替换为@ExcelProperty,如下图所示,其中value属性对应了我们的模板中的表头

对应Controller层的代码也需要做如下调整:

@Log(title = "用户管理", businessType = BusinessType.EXPORT)@RequiresPermissions("system:user:export")@PostMapping("/export")public void export(HttpServletResponse response, SysUser user){List<SysUser> list = userService.selectUserList(user);//ExcelUtil<SysUser> util = new ExcelUtil<SysUser>(SysUser.class);//util.exportExcel(response, list, "用户数据");try(InputStream resourceAsStream =  SysUserController.class.getClassLoader().getResourceAsStream("UserExportTemplate.xlsx");ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream()).withTemplate(resourceAsStream).inMemory(true).build()) {response.setContentType("application/vnd.ms-excel");String filename=  URLEncoder.encode("用户管理数据导出","UTF-8");response.setHeader("Content-disposition","attachment;filename="+filename+".xlsx");WriteSheet writeSheet = EasyExcel.writerSheet().build();FillConfig fillConfig = FillConfig.builder().forceNewRow(Boolean.TRUE).build();excelWriter.fill(list, fillConfig, writeSheet);Map<String, Object> map = MapUtils.newHashMap();// 自定义日期map.put("dataDate", new Date());excelWriter.fill(map, writeSheet);// 拿到原始的Workbook对象Workbook workbook = excelWriter.writeContext().writeWorkbookHolder().getWorkbook();// 获取已经填充的sheet对象Sheet userSheet = workbook.getSheetAt(0);for(Row row : userSheet){for (Cell cell : row){if(cell.getCellType() == CellType.STRING &&StringUtils.isNotBlank(cell.getStringCellValue())&& cell.getStringCellValue().contains("=")){cell.setCellFormula(cell.getStringCellValue().substring(1));}}}// 关闭excelWriter.finish();} catch (IOException e) {e.printStackTrace();}}

 我们首先获取到了用户信息的list对象, 随后加载自定义的模板 UserExportTemplate.xlsx ,使用EasyExcel 构建一个 WriteSheet 对象,其作用为填充自定义的日期对象以及我们的查询的list对象,待填充完成后。通过excelWriter再次获取到初始的WorkBook对象,在这里就是如何解决公式自动计算的关键,通过workBookt获取到Sheet对象后对其进行遍历,依次获取到Row和Cell对象, 这里的Cell对象就是我们的单元格,而我们填值的单元格为String类型,因此需要进行过滤,并且此处需要特别注意,去掉公式中原有的 = 符号,否则会引发异常。最后需要使用finish方法关闭流。该方案来源于官方文档easyexcel模板填充后,有公式的单元格不会自动计算,而需要我手动算,如何处理? | Easy Excel 官网

以及github上的 issue https://github.com/alibaba/easyexcel/issues/3242 

格式优化

虽然上述内容已经解决了公式导出后无法自动计算的问题,但是如果仔细观察就会发现有个问题可以进一步优化,由于最开始填入的单元格公式为A2+A3,而当增加了自定义表头后,此处内容应变为A5+A6,在Excel中这种情况会自动处理,但在当前场景下,只能通过手动方式去修正最终的单元格,而如果带有公式的单元格非常多时就会变得非常麻烦,因此可以通过如下代码进行调整:

 @Log(title = "用户管理", businessType = BusinessType.EXPORT)@RequiresPermissions("system:user:export")@PostMapping("/export")public void export(HttpServletResponse response, SysUser user){List<SysUser> list = userService.selectUserList(user);//ExcelUtil<SysUser> util = new ExcelUtil<SysUser>(SysUser.class);//util.exportExcel(response, list, "用户数据");try(InputStream resourceAsStream =  SysUserController.class.getClassLoader().getResourceAsStream("UserExportTemplate.xlsx");ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream()).withTemplate(resourceAsStream).inMemory(true).build()) {response.setContentType("application/vnd.ms-excel");String filename=  URLEncoder.encode("用户管理数据导出","UTF-8");response.setHeader("Content-disposition","attachment;filename="+filename+".xlsx");WriteSheet writeSheet = EasyExcel.writerSheet().build();FillConfig fillConfig = FillConfig.builder().forceNewRow(Boolean.TRUE).build();excelWriter.fill(list, fillConfig, writeSheet);Map<String, Object> map = MapUtils.newHashMap();// 自定义日期map.put("dataDate", new Date());excelWriter.fill(map, writeSheet);// 根据自定义表头调整公式int rowOffset = 3;// 拿到原始的Workbook对象Workbook workbook = excelWriter.writeContext().writeWorkbookHolder().getWorkbook();// 获取已经填充的sheet对象Sheet userSheet = workbook.getSheetAt(0);for(Row row : userSheet){for (Cell cell : row){if(cell.getCellType() == CellType.STRING &&StringUtils.isNotBlank(cell.getStringCellValue())&& cell.getStringCellValue().contains("=")){// 获取当前单元格内容String formula = cell.getStringCellValue();// 正则表达式匹配单元格引用 (例如 A1, B2 等)Pattern pattern = Pattern.compile("([A-Z]+)(\\d+)");Matcher matcher = pattern.matcher(formula);StringBuffer adjustedFormula = new StringBuffer();while(matcher.find()) {// 列名 (A, B, ...)String column = matcher.group(1);// 行号 (1, 2, ...)int rowNum = Integer.parseInt(matcher.group(2));// 调整行号int adjustedRow = rowNum + rowOffset;// 替换原始引用为调整后的引用matcher.appendReplacement(adjustedFormula, column + adjustedRow);}matcher.appendTail(adjustedFormula);// 重新设置公式cell.setCellFormula(adjustedFormula.substring(1));}}}// 关闭excelWriter.finish();} catch (IOException e) {e.printStackTrace();}}

此处通过在验证公式时使用正则匹配原有的公式,并根据设置的 rowOffset 动态进行调整(rowOffset为3是因为当前填入的公式从A2为基准开始,因此偏移量为3而并非4),使其能够正确计算结果。(此处代码也可抽取为方法在查询时list对象时直接使用)

最终效果

可以看到前端页面中填入的内容为=A2+A3,而导出文件内容为=A5+A6,并且打开导出的文件后自动计算了结果。

总结

对于具有自定义表头内容的导出需求,需要使用EasyExcel工具实现功能增强,并且需要先将内容写入到内容中之后,再读取将公式进行进一步过滤转换(去除等号,调整行号),进而实现导出文文件的计算功能,根据官方文档所述,该方案适用于文件小的场景,当文件过大时将可能造成内存溢出的问题,因此官方建议在Java层实现对公式的计算后再进行导出,如果各位读者有更好的想法欢迎评论区留言探讨。

相关文章:

EasyExcel模板导出与公式计算(下)

目录 环境要求 功能预览 需求分析 导入依赖 制作模板 编写代码 格式优化 最终效果 总结 在上一篇 EasyExcel模板导出与公式计算&#xff08;上&#xff09;-CSDN博客 文章中我们知道了在若依中使用自带的Excel注解来实现表格数据的导出&#xff0c;并且通过重写相关接…...

Golang序言全面学习-前序

最近看了很多与Golang有关的教程与书籍&#xff0c;也包括bilibili上的教程&#xff0c;各位老师讲解非常详细的&#xff0c;基本涉及了基础篇的方方面面&#xff0c;但总是感觉缺少了一些实战&#xff0c;以及实际经验的传授。实际项目会用到的日志框架、配置管理框架&#xf…...

Python世界:文件自动化备份实践

Python世界&#xff1a;文件自动化备份实践 背景任务实现思路坑点小结 背景任务 问题来自《简明Python教程》中的解决问题一章&#xff0c;提出实现&#xff1a;对指定目录做定期自动化备份。 最重要的改进方向是不使用 os.system 方法来创建归档文件&#xff0c; 而是使用 zip…...

PTA 6-10 阶乘计算升级版(详讲)

6-10 阶乘计算升级版 - 基础编程题目集 (pintia.cn)https://pintia.cn/problem-sets/14/exam/problems/type/6?problemSetProblemId742&page0 首先这道题不能用我们之前学过的阶乘计算方法来解决&#xff0c;比如下面这段代码就无法通过全部的样例 void Print_Factorial…...

软件开发人员从0到1实现物联网项目:项目架构的思考

文章目录 前言单体应用足矣摒弃传统的微信对接后期的维护投入上真正的“云”&#xff1a;云托管0服务器免运维免费的CDN和DDoS防护 技术架构小结 前言 因为种种原因&#xff0c;《软件开发人员从0到1实现物联网项目》这个项目的进度停滞了将近一个月。 鉴于该项目的前期开发和…...

Java--集合进阶 Collection,迭代器,lambda表达式

集合体系结构 Collection&#xff1a;单列集合 LIst系列集合&#xff1a;添加的元素是有序、可重复、有索引 Set系列集合&#xff1a;添加的元素是无序、不重复、无索引 Collection集合常用方法 | 方法名 | 说明 || :---…...

STM32G474之DAC

STM32G474分别使用CORDIC硬件和“math.h”的正弦值&#xff0c;从DAC1和DAC2输出。 1、DAC特点 PA4的附加功能为DAC1_OUT1&#xff0c;无需映射&#xff0c;直接将它配置为模拟功能&#xff0c;就可以使用了。 PA6的附加功能为DAC2_OUT1&#xff0c;无需映射&#xff0c;直接将…...

哈希表的底层实现(2)---C++版

目录 链地址法Separate Chaining——哈希桶的模拟实现 超大重点分析&#xff1a; 两种方法对比 由于在上次的哈希表的底层实现(1)---C版已经详细的阐述了相关的结构和原理&#xff0c;哈希表的实现方法主要分为链地址法和开放定址法。开放定址法上次已经实现过了&#xff0c…...

算法知识点————【LRU算法】

思想&#xff1a;淘汰最久没有使用的 应用场景&#xff1a;手机清后台的时候先清最久没有使用的应用 设计一种数据结构&#xff1a;接收一个 capacity 参数作为缓存的最大容量&#xff0c;然后实现两个 API&#xff0c;一个是 put(key, val) 方法存入键值对&#xff0c;另一个是…...

记一次MySQL视图查询优化的经验

背景&#xff1a;库房系统项目迁移&#xff0c;两个版本的结构发生了很大变化&#xff0c;新版本的库存系统在开发阶段由于数据量小&#xff0c;根据看不出查询的性能问题&#xff0c;还沾沾自喜的想新版本多好。但是在做同步之后&#xff08;规则变更&#xff0c;需要插入很多…...

Cloudways搭建WordPress外贸独立站完整教程(1)

验证邮件发送完成后&#xff0c;就等待Cloudways的回复邮件&#xff0c;一般24小时之内就会收到激活的邮件。 Cloudways账号升级 激活成功后还需要账户升级&#xff0c;Cloudways提供了为期3天的免费试用体验。如果在试用期结束之前未绑定信用卡以升级账户&#xff0c;试用期…...

Delphi5数据控制组件——查询

文章目录 效果图参考查询Free方法Close方法总结通俗理解 完整代码 效果图 参考 本文是在上一篇的基础上&#xff0c;将查询页面重新写一次。 查询 {点击查询} procedure TForm2.Button1Click(Sender: TObject); vartj,tj1,tj2,tj3,tj4,tj5,tj6,tj7:string; begin//按照工号查…...

git pull之后发现项目错误,如何回到之前的版本方法

目录 首先我们打开小程序的cmd的黑窗口&#xff0c;git reflog查看之前的版本 之后再git reset --hard main{1} 我这个就已经返回了之前的6daaa2e的版本了 首先我们打开小程序的cmd的黑窗口&#xff0c;git reflog查看之前的版本 之后再git reset --hard main{1} 我这个就已…...

防跌倒识别摄像机

防跌倒识别摄像机 是一种结合了人工智能技术和监控摄像技术的先进设备&#xff0c;旨在通过实时监测和分析监控画面中的行为动作&#xff0c;及时发现并预防跌倒事件的发生。这种摄像机在医疗、养老院、家庭等场所有着广泛的应用前景。 防跌倒识别摄像机在医疗领域具有重要意义…...

MyQql性能诊断与实践

获取更多免费资料&#xff0c;见下图...

有序序列判断

描述 输入一个整数序列&#xff0c;判断是否是有序序列&#xff0c;有序&#xff0c;指序列中的整数从小到大排序或者从大到小排序(相同元素也视为有序)。 数据范围&#xff1a;3 < n< 50 序列中的值都满足 1< val < 100 输入描述&#xff1a; 第一行输入一个整数N…...

【Kubernetes知识点问答题】健康检查

目录 1. Kubernetes 对集群 Pod 和容器健康状态如何进行监控和检测的。 2. 解释 LivenessProbes 探针的作用及其适用场景。 3. 解释 ReadinessProbe 探针的作用及其适用场景。 4. 解释 StartupProbe 探针的作用及其适用场景。 5. 说明 K8s 中 Pod 级别的 Graceful Shutdown…...

【Prometheus】PromQL数据类型以及常用的计算函数用法详解

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…...

STM32高级定时器生成互补PWM的原理与代码实现

文章目录 前言一 CubeMx配置1.1 TIM1 Mode and Configuration1.2 Paramter Settings 二 程序代码三 仿真分析总结 前言 互补 PWM&#xff08;Complementary PWM&#xff09;是指一对逻辑状态互为反相的 PWM&#xff08;脉冲宽度调制&#xff09;信号。这种信号配置常见于电机控…...

双指针题总结

双指针题总结 hot100移动零盛水最多的容器三数之和接雨水最小覆盖子串 hot100 移动零 题目链接&#xff1a; 283.移动零 代码&#xff1a; class Solution {public void moveZeroes(int[] nums) {int slow 0;for (int fast 0; fast < nums.length; fast ){if (nums[fas…...

告别信息混乱:Trilium中文版让知识管理像整理衣柜一样简单

告别信息混乱&#xff1a;Trilium中文版让知识管理像整理衣柜一样简单 【免费下载链接】trilium-translation Translation for Trilium Notes. Trilium Notes 中文适配, 体验优化 项目地址: https://gitcode.com/gh_mirrors/tr/trilium-translation 还在为英文笔记软件的…...

解决Gradio share=True报错:手动下载并配置frpc_linux_amd64_v0.3文件的保姆级教程

解决Gradio shareTrue报错的完整实战指南&#xff1a;从手动配置frpc到深度优化 当你兴奋地准备向客户展示刚完成的Gradio应用时&#xff0c;却在终端看到红色的报错信息——shareTrue参数失效了。这种场景对开发者来说再熟悉不过&#xff1a;本地调试一切正常&#xff0c;但需…...

Java全栈工程师面试实录:从基础到实战的深度技术探讨

Java全栈工程师面试实录&#xff1a;从基础到实战的深度技术探讨 一、面试开场 面试官&#xff08;李工&#xff09;&#xff1a;你好&#xff0c;欢迎来到我们公司。我是李工&#xff0c;负责技术面试。今天我们会围绕你的技术栈进行一些深入交流。 应聘者&#xff08;张明&am…...

从Gazebo到真实硬件:robot_state_publisher在ROS 2仿真迁移中的5个关键配置项

从Gazebo到真实硬件&#xff1a;robot_state_publisher在ROS 2仿真迁移中的5个关键配置项 当你在Gazebo中完成机器人运动算法的仿真验证后&#xff0c;下一步就是将这套系统部署到真实硬件上。这个过程中&#xff0c;robot_state_publisher的配置往往是工程师们最容易踩坑的环节…...

SiameseUIE多任务统一Schema设计:一套定义覆盖NER/关系/事件/情感

SiameseUIE多任务统一Schema设计&#xff1a;一套定义覆盖NER/关系/事件/情感 1. 引言&#xff1a;信息抽取的“瑞士军刀” 想象一下&#xff0c;你手头有一堆杂乱无章的中文文档——可能是新闻稿、用户评论、技术报告或者客服对话。老板让你快速从中找出所有提到的人名、公司…...

期权到期日别慌!手把手教你搞定上交所股票期权的行权与交割(附避坑清单)

期权到期日实战指南&#xff1a;从行权准备到交割避坑全流程解析 手机屏幕上的红色倒计时提醒着期权合约即将到期&#xff0c;作为刚接触期权交易不久的新手&#xff0c;此刻最需要的不再是复杂的概念解释&#xff0c;而是一份能握在手中的应急操作清单。本文将用最直白的语言拆…...

协方差矩阵可视化指南:如何用Seaborn热力图解读变量关系(附完整代码)

协方差矩阵可视化指南&#xff1a;如何用Seaborn热力图解读变量关系&#xff08;附完整代码&#xff09; 在数据分析的实际工作中&#xff0c;我们常常需要向非技术背景的决策者解释复杂的统计结果。这时候&#xff0c;一张直观的热力图往往比几十页的统计报告更有说服力。协方…...

我已战胜一切!感谢哥白尼,感谢爱因斯坦,感谢豆包,,,曾经我都经历过什么,我自己非常清楚,既有爱因斯坦的压缩版,又有哥白尼的压缩版,,,

不是时代不好&#xff0c;是人心中的成见就像一座大山般&#xff0c;无法被逾越&#xff0c;只有暴雨降下&#xff0c;洗刷这个世界&#xff0c;重塑这个宇宙&#xff0c;各位其位&#xff0c;大道至简。历史的车轮早已不可阻挡&#xff0c;&#xff0c;&#xff0c;暴风雨会来…...

cv2.findContours()错误的解决办法ValueError: not enough values to unpack (expected 3, got 2)

方法一&#xff1a;直接去掉一个返回值就即可。 方法二&#xff1a;把OpenCV 安装3.X的版本 具体原因 2、解析差异&#xff1a; OpenCV2和OpenCV4中&#xff1a; findContours这个轮廓提取函数会返回两个值&#xff1a;①轮廓的点集(contours)②各层轮廓的索引(hierarchy) 返回…...

苹果设备激活锁终极解锁指南:5步免费绕开iOS 15-16的iCloud限制

苹果设备激活锁终极解锁指南&#xff1a;5步免费绕开iOS 15-16的iCloud限制 【免费下载链接】applera1n icloud bypass for ios 15-16 项目地址: https://gitcode.com/gh_mirrors/ap/applera1n 还在为忘记Apple ID密码而无法使用自己的iPhone或iPad而烦恼吗&#xff1f;…...