官网:https://www.yuque.com/easyexcel/doc/easyexcel

导出

准备工作

引入依赖

<!--EasyExcel相关依赖-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>easyexcel</artifactId>
<version>3.0.5</version>
</dependency>

创建监听器

@Data
public class ExcelDataListener<T> extends AnalysisEventListener<T> {
private static Logger log4 = Logger.getLogger(AreaServiceImpl.class); /**
* 导入出错的信息(此处保存的为解析时错误的信息,某行某列+错误信息+;)
*/
private List<String> errorMsg = new ArrayList<>(); /**
* 必要列为空的信息(某行+'列1,列2...'+的值为空)
*/
private List<String> nullFields = new ArrayList<>(); /**
* 导入的表头(解析到的表头)
*/
private Map<Integer, String> headMap = new HashMap<>(); /**
* 缓存的数据(成功解析的数据)
*/
private List<T> dataList = new ArrayList<>(); /**
* 读取报错的数据(解析错误的数据,即无法映射到导出的实体类)
*/
private List<Map<String, String>> errorDataList = ListUtils.newArrayList(); /** 导出实体类的行索引字段名。
*用于设置解析到的数据所在的行数(在进行导入的时候用于定位service中save出错的数
*据,所在的行)
*/
private String rowIndexField; /**
* 假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。当然如果不用存储这个对象没用。
*/
// private DemoDAO demoDAO; // public ExcelDataListener(DemoDAO demoDAO) {
// this.demoDAO = demoDAO;
// } /**
* 这个每一条数据解析都会来调用
*
* @param data one row value
* @param context
*/
@Override
public void invoke(T data, AnalysisContext context) {
log4.info("解析到一条数据:" + JSON.toJSONString(data));
// 由于会读取空行(excel中删除数据后的行)所以这里加判断
if (!"{}".equals(JSON.toJSONString(data))) {
// 由于第一行是表头,因此这里+1
int rowIndex = context.readRowHolder().getRowIndex() + 1;
Class<?> dataClazz = data.getClass();
try {
// 如果没有设置实体类行索引的字段名,则默认为rowIndex
String fieldName = StringUtils.isEmpty(rowIndexField) ? "rowIndex" : rowIndexField;
Field rowIndexField = dataClazz.getDeclaredField(fieldName);
rowIndexField.setAccessible(true);
rowIndexField.set(data,rowIndex);
} catch (NoSuchFieldException | IllegalAccessException e) {
e.printStackTrace();
log4.info("对应的实体中没有行数");
}
// 校验必要列是否为空
List<String> checkField = ExcelUtil.checkField(data);
if (checkField.size() > 0) {
String errorMsg = "第" + rowIndex + "行," +
String.join("、",checkField) + "列的值为空";
nullFields.add(errorMsg);
} else {
dataList.add(data);
}
}
else
context.readRowHolder().setRowType(RowTypeEnum.EMPTY);
} /**
* 读取的表头
*
* @param headMap 表头
* @param context
*/
@Override
public void invokeHeadMap(Map<Integer, String> headMap, AnalysisContext context) {
// 此处由于会读取空单元格表头(删除内容后的单元格也会被读取)所以进行遍历判断
Set<Map.Entry<Integer, String>> entries = headMap.entrySet();
for (Map.Entry<Integer, String> head : entries) {
if (!StringUtils.isEmpty(head.getValue()))
this.headMap.put(head.getKey(),head.getValue());
}
} @Override
public boolean hasNext(AnalysisContext context) {
// 由于会读取空行数据,因此此处加判断遇到空行停止读取;(invoke方法中遇到空行会设置RowType)
if (RowTypeEnum.EMPTY.equals(context.readRowHolder().getRowType())) {
doAfterAllAnalysed(context);
return false;
}
return super.hasNext(context);
} /**
* 所有数据解析完成了 都会来调用
*
* @param context
*/
@Override
public void doAfterAllAnalysed(AnalysisContext context) {
// 这里也要保存数据,确保最后遗留的数据也存储到数据库
log4.info("所有数据解析完成!");
} /**
* 在转换异常 获取其他异常下会调用本接口。抛出异常则停止读取。如果这里不抛出异常则 继续读取下一行。
*
* @param exception
* @param context
* @throws Exception
*/
@Override
public void onException(Exception exception, AnalysisContext context) {
// 如果是某一个单元格的转换异常 能获取到具体行号
// 如果要获取头的信息 配合invokeHeadMap使用
if (exception instanceof ExcelDataConvertException) {
ExcelDataConvertException e = (ExcelDataConvertException)exception;
String errorIndexInfo = "第" + (e.getRowIndex() + 1) + "行"
+ "的”" +headMap.get(e.getColumnIndex()) +"“列,";
errorMsg.add(errorIndexInfo + exception.getMessage());
// 构造错误的数据
ReadRowHolder readRowHolder = context.readRowHolder();
HashMap<String, String> data = new HashMap<>();
Set<Map.Entry<Integer, Cell>> entries = readRowHolder.getCellMap().entrySet();
for (Map.Entry<Integer, Cell> entry : entries) {
ReadCellData value = (ReadCellData)(entry.getValue());
data.put(headMap.get(entry.getKey()),value.getStringValue());
}
errorDataList.add(data);
}
} }

工具类校验必要字段

/**
* 校验导入实体的必要字段
* @param entity 导入的实体
* @param <T>
* @return
*/
public static <T> List<String> checkField(T entity){
Class<?> clazz = entity.getClass();
Field[] fields = clazz.getDeclaredFields();
List<String> nullFields = new ArrayList<>();
for (Field field : fields) {
NeedNotNull notNull = field.getDeclaredAnnotation(NeedNotNull.class);
ExcelProperty property = field.getDeclaredAnnotation(ExcelProperty.class);
field.setAccessible(true);
// 该字段是否不能为空
if (!ObjectUtils.isEmpty(notNull)){
try {
Object value = field.get(entity);
// 如果必要字段的值为空则校验失败
if (ObjectUtils.isEmpty(value))
nullFields.add(StringUtils.join(property.value(),","));
} catch (IllegalAccessException e) {
e.printStackTrace();
log4.error(field.getName() + "字段值获取异常:"+e.getMessage());
}
}
}
return nullFields;
}

自定义注解

/**
* 自定义注解,用于判断是否需要合并以及合并的主键
*/
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface CustomMerge { /**
* 是否需要合并单元格
*/
boolean needMerge() default false; /**
* 是否是主键,即该字段相同的行合并
*/
boolean isPk() default false;
} /**
* 自定义注解,用于判断字段是否不能为空
*/
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface NeedNotNull { /**
* 是否不能为空
*/
boolean value() default true;
/**
* 字段名称
*/
String name() default "";
}

创建导出的实体类

@Data
public class AreaImportDto { /** 区域名称 */
@ExcelProperty("区域名称")
@CustomMerge(needMerge = true, isPk = true)
@ColumnWidth(20)
@NeedNotNull(name = "区域名称")
private String areaName; /** 区域的X坐标 */
@ExcelProperty("父区域")
@CustomMerge(needMerge = true)
@ColumnWidth(10)
private String parentName; /** 人员名称 */
@ExcelProperty("人员名称")
@CustomMerge(needMerge = true)
@ColumnWidth(20)
private String personName; /** 导入数据所在的行数 */
private Integer rowIndex;
}

ps:

@ExcelProperty: 导出的列名(如果是多行表头,则值为数组,详见官网)

@CustomMerge(needMerge = true, isPk = true):要合并的列

@ColumnWidth(20):列宽

@NeedNotNull(name = "区域名称"):是否为必要列

构造数据

  • 调用service查询数据库数据
  • 定义转换方法,将实体类转换为导出实体(此处可以省略,如果不另外定义导出实体的话耦合性较高)

创建导出工具类

private static final String FILE_SEPARATOR = "/";

/**
* 数据导出到excel
*
* @param params 请求参数(自定义的请求参数实体,包含query(Map)、sort、page属性)
* @param serverPath 服务器文件导出路径
* @param defaultFileName 默认文件名
* @param targetData 导出数据
* @param clazz 导出数据的类对象
* @param isMergeStrategy 是否要合并行(一对多导出时用到)
* @return
* @throws IOException
*/
public static <T> Map<String, String> exportExcel(SearchParamDto params, String defaultFileName,
String serverPath, String urlPath, List<T> targetData,
Class<T> clazz, boolean isMergeStrategy) throws IOException {
HashMap<String, String> result = new HashMap<String, String>() {{
put("msg", "数据导出成功");
}};
// 请求中传递的需要导出的列
JSONArray fields = JSONArray.parseArray(params.getQuery().get("fields").toString());
// 设置文件名
String fileName1 = ObjectUtils.isEmpty(params.getQuery().get("fileName")) ?
defaultFileName :
params.getQuery().get("fileName").toString();
// String fileName = URLEncoder.encode(fileName1, "UTF-8").replaceAll("\\+", "%20") + ".xlsx";
String fileName = fileName1 + ".xlsx"; try {
// 获取当前时间
String updTm = DateUtil.getAllTime();
String filePath = serverPath + FILE_SEPARATOR + updTm + FILE_SEPARATOR + fileName;
String destDirName = serverPath + FILE_SEPARATOR + updTm;
createDir(destDirName);
FileOutputStream fileOutputStream = new FileOutputStream(filePath);
log4.info(fileName1 + "数据导出成功!");
ExcelWriterBuilder head = EasyExcel.write(fileOutputStream).autoCloseStream(true)
.includeColumnFiledNames(fields.toJavaList(String.class))
.head(clazz);
if (isMergeStrategy) {
head.registerWriteHandler(new CustomMergeStrategy(clazz));
}
head.excelType(ExcelTypeEnum.XLSX)
.sheet("sheet1")
.doWrite(targetData);
String url = urlPath + FILE_SEPARATOR + updTm + FILE_SEPARATOR + fileName;
// 返回请求文件的url以及服务器报错路径
result.put("url", url);
result.put("savePath", serverPath + FILE_SEPARATOR + updTm);
} catch (Exception e) {
log4.error("数据导出失败: {}", e);
result.put("msg", "数据导出失败");
return result;
} return result;
}

EasyExcel实现文件导出的更多相关文章

  1. SpringBoot基于EasyExcel解析Excel实现文件导出导入、读取写入

    1. 简介   Java解析.生成Excel比较有名的框架有Apache poi.jxl.但他们都存在一个严重的问题就是非常的耗内存,poi有一套SAX模式的API可以一定程度的解决一些内存溢出的问题 ...

  2. 使用Layui、Axios、Springboot(Java) 实现EasyExcel的导入导出(浏览器下载)

    实现EasyExcel的导入导出(浏览器下载) 实现三个按钮的功能,但是却花费了一天的时间包括总结. 使用到的技术:springboot layui axios EasyExcel mybatis-p ...

  3. 使用VUE+SpringBoot+EasyExcel 整合导入导出数据

    使用VUE+SpringBoot+EasyExcel 整合导入导出数据 创建一个普通的maven项目即可 项目目录结构 1 前端 存放在resources/static 下 index.html &l ...

  4. C#+OpenGL+FreeType显示3D文字(1) - 从TTF文件导出字形贴图

    C#+OpenGL+FreeType显示3D文字(1) - 从TTF文件导出字形贴图 +BIT祝威+悄悄在此留下版了个权的信息说: 最近需要用OpenGL绘制文字,这是个很费时费力的事.一般的思路就是 ...

  5. loadrunner实现excel文件导出操作

    项目中需要对“商品信息”进行查询及导出,但是loadrunner并不能录制到“保存”这一操作. 项目介绍:flex+Http协议: 不能录制的原因: 在我们点击了“导出”按钮后,服务端已经生成一份我们 ...

  6. C#进行Visio二次开发之文件导出及另存Web页面

    在我前面很多关于Visio的开发过程中,介绍了各种Visio的C#开发应用场景,包括对Visio的文档.模具文档.形状.属性数据.各种事件等相关的基础处理,以及Visio本身的整体项目应用,虽然时间过 ...

  7. Ado.Net小练习01(数据库文件导出,导入)

    数据库文件导出主要程序: <span style="font-family: Arial, Helvetica, sans-serif;"><span style ...

  8. JAVA实用案例之文件导出(JasperReport踩坑实录)

    写在最前面 想想来新公司也快五个月了,恍惚一瞬间. 翻了翻博客,因为太忙,也有将近五个多月没认真总结过了. 正好趁着今天老婆出门团建的机会,记录下最近这段时间遇到的大坑-JasperReport. 六 ...

  9. 文件导出也可以这么写【js+blob】

    文件导出在软件开发中是个比较常用的功能,基本原理也很简单: 浏览器向后台发送一个Get请求 后台处理程序接收到请求后,经过处理,返回二进制文件流 浏览器接收到二进制文件流后提示下载文件 调用的js方法 ...

随机推荐

  1. Operating System_via牛客网

    题目 链接:https://ac.nowcoder.com/acm/contest/28537/F 来源:牛客网 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 131072K,其他语 ...

  2. Python3.7+Django2.0.4配合Mongodb打造高性能高扩展标签云存储方案

    原文转载自「刘悦的技术博客」https://v3u.cn/a_id_141 书接上回,之前有一篇文章提到了标签云系统的构建:Python3.7+jieba(结巴分词)配合Wordcloud2.js来构 ...

  3. 内存问题难定位,那是因为你没用ASAN

    摘要:ASAN全称:Address Sanitizer,google发明的一种内存地址错误检查器.目前已经被集成到各大编译器中. 本文分享自华为云社区<内存定位利器-ASAN使用小结>,作 ...

  4. 跳转语句break、continue、return

    1.break 语句 概念: break语句在循环中的作用是终止当前循环,在switch语句中的作用是终止switch. 示例: 输出结果:  2.continue 语句 概念: continue语句 ...

  5. LeetCode使用JavaScript破解两数之和

    有人相爱,有人夜里开车看海,我是leetcode第一题都做不出来 题目 给定一个整数数组 nums 和一个整数目标值 target,请你在该数组中找出 和为目标值 target 的那 两个 整数,并返 ...

  6. 为MySQL MGR实现简单的负载均衡代理

    GreatSQL社区原创内容未经授权不得随意使用,转载请联系小编并注明来源. 原创:万里数据库,花家舍 导读 在多写(多节点写入)数据库(例如MySQL MGR的multi-primary mode) ...

  7. ASP.NET Core 6框架揭秘实例演示[32]:错误页面的集中呈现方式

    由于ASP.NET是一个同时处理多个请求的Web应用框架,所以在处理某个请求过程中出现异常并不会导致整个应用的中止.出于安全方面的考量,为了避免敏感信息外泄,客户端在默认情况下并不会得到详细的出错信息 ...

  8. Bert不完全手册7. 为Bert注入知识的力量 Baidu-ERNIE & THU-ERNIE & KBert

    借着ACL2022一篇知识增强Tutorial的东风,我们来聊聊如何在预训练模型中融入知识.Tutorial分别针对NLU和NLG方向对一些经典方案进行了分类汇总,感兴趣的可以去细看下.这一章我们只针 ...

  9. 管理 MongoDB 用户和权限

    创建用户 创建用户的函数是:db.createUser(). 创建用户时,需要为该用户添加权限.可添加的权限以及说明: 权限 作用 read 允许用户读取指定数据库. readWrite 允许用户读写 ...

  10. Matery主题添加Pjax

    如何给matery主题添加Pjax? Pjax优点 1.减轻服务端压力 2.按需请求,每次只需加载页面的部分内容,而不用重复加载一些公共的资源文件和不变的页面结构,大大减小了数据请求量,以减轻对服务器 ...