SpringBoot 数据库导入导出 Xlsx文件的导入与导出 全量导出 数据库导出表格 数据处理 外部数据

article/2025/6/8 10:44:47

介绍

poi-ooxml 是 Apache POI 项目中的一个库,专门用于处理 Microsoft Office 2007 及以后版本的文件,特别是 Excel 文件(.xlsx 格式)和 Word 文件(.docx 格式)。

在管理系统中需要对数据库的数据进行导入或导出在系统中经常使用的到。

Java针对MS Office的操作的库屈指可数,比较有名的就是Apache的POI库。这个库异常强大,但是使用起来也并不容易。Hutool针对POI封装一些常用工具,使Java操作Excel等文件变得异常简单。

胡图官网:https://doc.hutool.cn/pages/poi/#%E7%94%B1%E6%9D%A5

Hutool-poi是针对Apache POI的封装,因此需要用户自行引入POI库,Hutool默认不引入。

依赖

<dependency><groupId>org.apache.poi</groupId><artifactId>poi-ooxml</artifactId><version>4.1.2</version>
</dependency><dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.7.18</version>
</dependency>

实体类

支持别名注解的。可以在字段上加@Alias注解。

@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@TableName("site")
public class Site implements Serializable {private static final long serialVersionUID = 1L;@TableId(value = "id", type = IdType.AUTO)@Alias(value = "编号")private Integer id;@TableField("ip_address")@Alias(value = "IP地址")private String ipAddress;@TableField("source")@Alias(value = "来源")private String source;@TableField("operation")private String operation;@TableField("access_time")private LocalDateTime accessTime;}

在这里插入图片描述

导出Xls

 @GetMapping()public void export(HttpServletResponse response) throws Exception{String fileName = "测试.xls";String encodedFileName = URLEncoder.encode(fileName, "UTF-8");encodedFileName = encodedFileName.replace("+", "%20");  // 处理空格字符List<Site> list = siteService.list();
// 通过工具类创建writer,默认创建xls格式ExcelWriter writer = ExcelUtil.getWriter();// 一次性写出内容,使用默认样式,强制输出标题writer.write(list, true);//response为HttpServletResponse对象response.setContentType("application/vnd.ms-excel;charset=utf-8");//test.xls是弹出下载对话框的文件名response.setHeader("Content-Disposition","attachment;filename="+encodedFileName);ServletOutputStream out=response.getOutputStream();writer.flush(out, true);// 关闭writer,释放内存writer.close();
//此处记得关闭输出Servlet流IoUtil.close(out);}

在这里插入图片描述

导出Xlsx

@GetMapping()
public void export(HttpServletResponse response) throws Exception{String  fileName = URLEncoder.encode("测试.xlsx", "UTF-8").replace("+", "%20");List<Site> list = siteService.list();ExcelWriter writer = ExcelUtil.getWriter(true);writer.write(list, true);response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8");response.setHeader("Content-Disposition","attachment;filename="+fileName);ServletOutputStream out=response.getOutputStream();writer.flush(out, true);writer.close();IoUtil.close(out);}

以上都是基于全量数据的导出方法,下面是按需导出。


创建自定义注解

@Retention(RetentionPolicy.RUNTIME) // 保证注解在运行时可访问
public  @interface ExcelTitle {String value(); // 用来保存 Excel 列标题
}

自定义实体类

@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@TableName("site")
public class Site implements Serializable {private static final long serialVersionUID = 1L;@TableId(value = "id", type = IdType.AUTO)private Integer id;@TableField("ip_address")@ExcelTitle("IP地址")private String ipAddress;@TableField("source")@ExcelTitle("来源")private String source;@TableField("operation")@ExcelTitle("测试字段")private String operation;@TableField("access_time")@ExcelTitle("访问时间")private LocalDateTime accessTime;}

按需不排序

@GetMapping() 
public void export(HttpServletResponse response) throws Exception{// 需要导出的字段名String[] key = {"访问时间", "IP地址", "来源"};// 查找性能较高Set<String> keySet = new HashSet<>(Arrays.asList(key));// 对文件名进行 URL 编码,防止中文字符出现乱码,并将 "+" 替换成 "%20"String fileName = URLEncoder.encode("测试.xlsx", "UTF-8").replace("+", "%20");// 从服务层获取数据列表,通常是从数据库或其他数据源获取List<Site> list = siteService.list();// 创建一个 ExcelWriter 实例,用于写入 Excel 文件,true 表示创建时有表头ExcelWriter writer = ExcelUtil.getWriter(true);// 获取 Site 类的所有字段(反射机制)Field[] fields = Site.class.getDeclaredFields();// 遍历每个字段for (Field field : fields) {// 获取字段上是否有 @ExcelTitle 注解ExcelTitle excelAnnotation = field.getAnnotation(ExcelTitle.class);// 如果字段上存在 @ExcelTitle 注解if (excelAnnotation != null) {// 获取注解中的 value 值(即字段对应的 Excel 标题)String excelValue = excelAnnotation.value();// 判断当前字段的 Excel 标题是否是我们关心的字段if (Arrays.asList(key).contains(excelValue)) {// 如果是我们关心的字段,则添加标题别名(即字段名与 Excel 表头的映射关系)writer.addHeaderAlias(field.getName(), excelValue);}}}// 默认情况下,未添加 alias 的属性也会被写出。如果我们只想输出加了别名的字段,可以调用该方法writer.setOnlyAlias(true);// 将数据写入 Excel 文件,true 表示需要写入表头writer.write(list, true);// 设置响应内容类型为 Excel 文件格式response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8");// 设置响应头,指定下载的文件名response.setHeader("Content-Disposition", "attachment;filename=" + fileName);// 获取响应输出流,用于将文件写入客户端ServletOutputStream out = response.getOutputStream();// 将 Excel 内容刷新到输出流writer.flush(out, true);// 关闭 writer 释放资源writer.close();// 关闭输出流IoUtil.close(out);
}

按需排序导出

    @GetMapping()public void export(HttpServletResponse response) throws Exception {String[] key = {"访问时间", "IP地址", "来源"};Set<String> keySet = new HashSet<>(Arrays.asList(key));String fileName = URLEncoder.encode("测试.xlsx", "UTF-8").replace("+", "%20");List<Site> list = siteService.list();ExcelWriter writer = ExcelUtil.getWriter(true);for (String k : key) {String title =  getCharacter(Site.class, k);if(title==null)continue;writer.addHeaderAlias(title, k);}//        //自定义标题别名
//        writer.addHeaderAlias("ipAddress", "IP地址");// 默认的,未添加alias的属性也会写出,如果想只写出加了别名的字段,可以调用此方法排除之writer.setOnlyAlias(true);writer.write(list, true);response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet;charset=utf-8");response.setHeader("Content-Disposition", "attachment;filename=" + fileName);ServletOutputStream out = response.getOutputStream();writer.flush(out, true);writer.close();IoUtil.close(out);}
/*** 根据传入的类和关键字,查找类中具有 @ExcelTitle 注解的字段,* 如果该字段的 @ExcelTitle 注解的 value 属性与传入的关键字匹配,则返回该字段的名称。** @param clazz 目标类的 Class 对象,用于获取类的字段信息。* @param key   需要匹配的关键字,用于查找与其对应的 @ExcelTitle 注解的字段。* @return 如果找到匹配的字段,则返回字段名称;否则返回 null。*/
public String getCharacter(Class<Site> clazz, String key) {// 获取目标类的所有字段Field[] fields = clazz.getDeclaredFields();// 遍历所有字段for (Field field : fields) {// 获取当前字段的 @ExcelTitle 注解ExcelTitle excelAnnotation = field.getAnnotation(ExcelTitle.class);// 如果字段具有 @ExcelTitle 注解if (excelAnnotation != null) {// 获取注解的 value 值String excelValue = excelAnnotation.value();// 检查传入的 key 是否与 @ExcelTitle 注解中的值匹配if (Arrays.asList(key).contains(excelValue)) {// 如果匹配,返回字段名称return field.getName();}}}// 如果没有找到匹配的字段,返回 nullreturn null;
}

在这里插入图片描述


http://www.hkcw.cn/article/CrybxNwFFZ.shtml

相关文章

EagleTrader采访|在市场中修行的交易之道与实战反思

在交易世界的洪流中&#xff0c;真正让人敬佩的不是一时的高收益&#xff0c;而是对规则的敬畏与长期自我修正的能力。来自EagleTrader的交易员罗秋林&#xff0c;正是这样一位在实战中历练出的“老兵”。 七八年时间&#xff0c;他从爆仓阴影走出&#xff0c;如今选择通过Eag…

OpenCV CUDA模块霍夫变换------在 GPU 上执行概率霍夫变换检测图像中的线段端点类cv::cuda::HoughSegmentDetector

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 cv::cuda::HoughSegmentDetector 是 OpenCV 的 CUDA 模块中一个非常重要的类&#xff0c;它用于在 GPU 上执行 概率霍夫变换&#xff08;Probabi…

MybatisPlus--核心功能--service接口

Service接口 基本用法 MyBatisPlus同时也提供了service接口&#xff0c;继承后一些基础的增删改查的service代码&#xff0c;也不需要去书写。 接口名为Iservice&#xff0c;而Iservice也继承了IRepository&#xff0c;这里提供的方法跟BaseMapper相比只多不少&#xff0c;整…

Silky-CTF: 0x02靶场

Silky-CTF: 0x02 来自 <Silky-CTF: 0x02 ~ VulnHub> 1&#xff0c;将两台虚拟机网络连接都改为NAT模式 2&#xff0c;攻击机上做namp局域网扫描发现靶机 nmap -sn 192.168.23.0/24 那么攻击机IP为192.168.23.128&#xff0c;靶场IP192.168.23.131 3&#xff0c;对靶机进…

图片组件|纯血鸿蒙组件库AUI

摘要&#xff1a; 图片组件(A_Image)&#xff1a;可设置图片地址、图片宽度、图片高度、圆角类型及是否显示外框线。圆角类型支持普通圆角、圆形及无圆角。 一、组件调用方式 1.极简调用 只需要输入A_Image&#xff0c;然后给src&#xff08;图片地址&#xff09;属性赋值即可…

tGSSA-LSTM多输入回归|改进麻雀优化-长短期记忆神经网络|Matlab回归通用

目录 一、程序及算法内容介绍&#xff1a; 基本内容&#xff1a; 亮点与优势&#xff1a; 二、实际运行效果&#xff1a; 三、方法原理介绍&#xff1a; 四、完整程序下载&#xff1a; 一、程序及算法内容介绍&#xff1a; 基本内容&#xff1a; 本代码基于Matlab平台编译…

【吾爱】逆向实战crackme160破解记录(二)

前言 最近在拿吾爱上的crackme程序练练手&#xff0c;发现论坛上已经有pk8900总结好的160个crackme&#xff0c;非常方便&#xff0c;而且有很多厉害的前辈已经写好经验贴和方法了&#xff0c;我这里只是做一下自己练习的记录&#xff0c;欢迎讨论学习&#xff0c;感谢吾爱论坛…

C# Onnx 动漫人物人脸检测

目录 效果 模型信息 项目 代码 下载 参考 效果 模型信息 Model Properties ------------------------- stride&#xff1a;32 names&#xff1a;{0: face} --------------------------------------------------------------- Inputs ------------------------- name&am…

使用cmd命令行创建数据库和表-简单步骤记录

前提&#xff1a; 已安装MySQL 步骤&#xff1a; 1.WinR&#xff0c;回车&#xff0c;输入cmd&#xff0c;回车 2.输入 mysql -u root -p 后&#xff0c;输入自己的密码&#xff0c;看到welcome等字样就是成功登录了MySQL 3.创建数据库 create database success; &#xff0…

Centos7使用rpm升级glibc2.28

Centos7使用rpm升级glibc2.28 检查glibc版本下载glibc2.28的rpm包使用rpm包升级到glibc-2.28结果验证 检查glibc版本 ldd --version下载glibc2.28的rpm包 参考&#xff1a; https://www.cnblogs.com/caya-yuan/p/10561439.html 下载 glibc、make 的 feroda29(fc29)系统 rpm包…

堆叠弹窗 VS 队列弹窗之争

前言 如果一个页面上有多个弹窗&#xff0c;设计上是把前一个弹窗暂时隐藏还是盖住前一个弹窗多一点&#xff1f; 在多弹窗设计的情境下&#xff0c;最佳实践通常倾向于以下两种处理方式&#xff1a; 1、堆叠弹窗 新弹窗覆盖旧弹窗&#xff0c;但每个弹窗保持完整显示&#…

刷leetcode hot100返航必胜版--链表6/3

链表初始知识 链表种类&#xff1a;单链表&#xff0c;双链表&#xff0c;循环链表 链表初始化 struct ListNode{ int val; ListNode* next; ListNode(int x): val&#xff08;x&#xff09;,next(nullptr) {} }; //初始化 ListNode* head new ListNode(5); 删除节点、添加…

[概率论基本概念4]什么是无偏估计

关键词&#xff1a;Unbiased Estimation 一、说明 对于无偏和有偏估计&#xff0c;需要了解其叙事背景&#xff0c;是指整体和抽样的关系&#xff0c;也就是说整体的叙事是从理论角度的&#xff0c;而估计器原理是从实践角度说事&#xff1b;为了表明概率理论&#xff08;不可…

React-native之Flexbox

本文总结: 我们学到了 React Native 的 Flexbox 布局&#xff0c;它让写样式变得更方便啦&#xff01;&#x1f60a; Flexbox 就像一个有弹性的盒子&#xff0c;有主轴和交叉轴&#xff08;行或列&#xff09;。 在 RN 里写样式要用 StyleSheet.create 对象&#xff0c;属性名…

学习日记-day21-6.3

完成目标&#xff1a; 目录 知识点&#xff1a; 1.集合_哈希表存储过程说明 2.集合_哈希表源码查看 3.集合_哈希表无索引&哈希表有序无序详解 4.集合_TreeSet和TreeMap 5.集合_Hashtable和Vector&Vector源码分析 6.集合_Properties属性集 7.集合_集合嵌套 8.…

ABP-Book Store Application中文讲解 - Part 6: Authors: Domain Layer

ABP-Book Store Application中文讲解 - Part 6: Authors: Domain Layer 1. 汇总 ABP-Book Store Application中文讲解-汇总-CSDN博客 2. 前一章 ABP-Book Store Application中文讲解 - Part 5: Authorization-CSDN博客 项目之间的引用关系。 ​ BookAppService利用的是Cu…

智慧高铁站:数字时代交通枢纽的标杆

智慧高铁站作为现代综合交通体系的核心节点&#xff0c;通过数字技术与基础设施的深度融合&#xff0c;正在重塑旅客出行体验与车站运营模式。这一转型不仅体现在技术应用层面&#xff0c;更代表着交通服务理念的根本性变革&#xff0c;为现代交通枢纽建设树立了全新标杆。 一、…

ARM架构推理Stable Diffusiond

代码仓库&#xff1a; https://github.com/siutin/stable-diffusion-webui-docker.git Docker容器地址&#xff1a; https://hub.docker.com/r/siutin/stable-diffusion-webui-docker/tags git clone https://github.com/siutin/stable-diffusion-webui-docker.git cd stabl…

关于 KWDB 数据存储的几件事儿

邻近粽子节&#xff0c;KWDB 的朋友给我发消息&#xff0c;问我吃过红茶味的粽子没&#xff0c;作为北方人的我一般只吃蜜枣白粽&#xff0c;还没见过茶香粽子&#xff0c;顶多泡碗祁红&#xff0c;就着茶水吃粽子。 她又问道&#xff0c;两个月时间到了&#xff0c;你准备好了…

酵母杂交那些事儿(一)

酵母单杂、酵母双杂、酵母三杂&#xff0c;仅仅一个字的区别&#xff0c;你对它们了解吗&#xff1f;这些经常用到的实验&#xff0c;它们的原理你确定都搞清楚了吗&#xff1f;如果没有&#xff0c;那么今天你就来对地方了&#xff0c;因为伯远生物&#xff08;https://plant.…