• EasyExcel 优雅实现 Excel 导入导出


    图片

    一、简介

    EasyExcel是一个基于Java的、快速、简洁、解决大文件内存溢出的Excel处理工具。他能让你在不用考虑性能、内存的等因素的情况下,快速完成Excel的读、写等功能。

    二、特点

    快速

    快速的读取excel中的数据。

    简洁

    映射excel和实体类,让代码变的更加简洁。

    节约内存

    在读写大文件的时候使用磁盘做缓存,更加的节约内存。

    图片

    16M内存23秒读取75M(46W行25列)的Excel(3.2.1+版本)

    当然还有极速模式能更快,但是内存占用会在100M多一点。

    三、对比

    Java解析、生成Excel比较有名的框架有Apache poi、jxl。但他们都存在一个严重的问题就是非常的耗内存,poi有一套SAX模式的API可以一定程度的解决一些内存溢出的问题,但POI还是有一些缺陷,比如07版Excel解压缩以及解压后存储都是在内存中完成的,内存消耗依然很大。

    easyexcel重写了poi对07版Excel的解析,一个3M的excel用POI sax解析依然需要100M左右内存,改用easyexcel可以降低到几M,并且再大的excel也不会出现内存溢出;03版依赖POI的sax模式,在上层做了模型转换的封装,让使用者更加简单方便。

    四、注解

    @ExcelProperty

    用于匹配excel和实体类的匹配,参数如下:

    名称默认值描述
    value用于匹配excel中的头,必须全匹配,如果有多行头,会匹配最后一行头
    orderInteger.MAX_VALUE优先级高于value,会根据order的顺序来匹配实体和excel中数据的顺序
    index-1优先级高于valueorder,会根据index直接指定到excel中具体的哪一列
    converter自动选择指定当前字段用什么转换器,默认会自动选择。读的情况下只要实现com.alibaba.excel.converters.Converter#convertToJavaData(com.alibaba.excel.converters.ReadConverterContext) 方法即可

    @ExcelIgnore

    默认所有字段都会和excel去匹配,加了这个注解会忽略该字段

    @ExcelIgnoreUnannotated

    默认不加ExcelProperty 的注解的都会参与读写,加了不会参与读写

    @DateTimeFormat

    日期转换,用String去接收excel日期格式的数据会调用这个注解,参数如下:

    名称默认值描述
    value参照java.text.SimpleDateFormat书写即可
    use1904windowing自动选择excel中时间是存储1900年起的一个双精度浮点数,但是有时候默认开始日期是1904,所以设置这个值改成默认1904年开始

    @NumberFormat

    数字转换,用String去接收excel数字格式的数据会调用这个注解。

    名称默认值描述
    value参照java.text.DecimalFormat书写即可
    roundingModeRoundingMode.HALF_UP

    格式化的时候设置舍入模

    @ColumnWidth

    用于设置表格列的宽度(value = 20);

    五、版本选择

    版本poi依赖版本 (支持范围)jdk版本支持范围备注
    3.1.0+4.1.2 (4.1.2 - 5.2.2)jkd8 - jdk17推荐使用,会更新的版本
    3.0.0-beta1 - 3.0.54.1.2 (4.1.2 - 5.2.2)jkd8 - jdk11不推荐项目新引入此版本,除非超级严重bug,否则不再更新
    2.0.0-beta1-2.2.113.17 (3.17 - 4.1.2)jdk6 - jdk11不推荐项目新引入此版本,除非是jdk6否则不推荐使用,除非超级严重bug,否则不再更新
    1+版本3.17 (3.17 - 4.1.2)jdk6 - jdk11不推荐项目新引入此版本,超级严重bug,也不再更新

    六、使用示例

    1. 读Excel

    对象

    1. @Getter
    2. @Setter
    3. @EqualsAndHashCode
    4. public class DemoData {
    5. /**
    6. * 强制读取第三个 这里不建议 index 和 name 同时用,要么一个对象只用index,要么一个对象只用name去匹配
    7. */
    8. @ExcelProperty(index = 2)
    9. private Double doubleData;
    10. /**
    11. * 用名字去匹配,这里需要注意,如果名字重复,会导致只有一个字段读取到数据
    12. */
    13. @ExcelProperty("字符串标题")
    14. private String string;
    15. @ExcelProperty("日期标题")
    16. private Date date;
    17. }

    ​​​​​​​监听器

    1. // 有个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去
    2. @Slf4j
    3. public class DemoDataListener implements ReadListener {
    4. /**
    5. * 每隔5条存储数据库,实际使用中可以100条,然后清理list ,方便内存回收
    6. */
    7. private static final int BATCH_COUNT = 100;
    8. /**
    9. * 缓存的数据
    10. */
    11. private List cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
    12. /**
    13. * 假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。当然如果不用存储这个对象没用。
    14. */
    15. private DemoDAO demoDAO;
    16. public DemoDataListener() {
    17. // 这里是demo,所以随便new一个。实际使用如果到了spring,请使用下面的有参构造函数
    18. demoDAO = new DemoDAO();
    19. }
    20. /**
    21. * 如果使用了spring,请使用这个构造方法。每次创建Listener的时候需要把spring管理的类传进来
    22. *
    23. * @param demoDAO
    24. */
    25. public DemoDataListener(DemoDAO demoDAO) {
    26. this.demoDAO = demoDAO;
    27. }
    28. /**
    29. * 这个每一条数据解析都会来调用
    30. *
    31. * @param data one row value. Is is same as {@link AnalysisContext#readRowHolder()}
    32. * @param context
    33. */
    34. @Override
    35. public void invoke(DemoData data, AnalysisContext context) {
    36. log.info("解析到一条数据:{}", JSON.toJSONString(data));
    37. cachedDataList.add(data);
    38. // 达到BATCH_COUNT了,需要去存储一次数据库,防止数据几万条数据在内存,容易OOM
    39. if (cachedDataList.size() >= BATCH_COUNT) {
    40. saveData();
    41. // 存储完成清理 list
    42. cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
    43. }
    44. }
    45. /**
    46. * 所有数据解析完成了 都会来调用
    47. *
    48. * @param context
    49. */
    50. @Override
    51. public void doAfterAllAnalysed(AnalysisContext context) {
    52. // 这里也要保存数据,确保最后遗留的数据也存储到数据库
    53. saveData();
    54. log.info("所有数据解析完成!");
    55. }
    56. /**
    57. * 加上存储数据库
    58. */
    59. private void saveData() {
    60. log.info("{}条数据,开始存储数据库!", cachedDataList.size());
    61. demoDAO.save(cachedDataList);
    62. log.info("存储数据库成功!");
    63. }
    64. }

    代码

    1. /**
    2. * 指定列的下标或者列名
    3. *
    4. *

      1. 创建excel对应的实体对象,并使用{@link ExcelProperty}注解. 参照{@link IndexOrNameData}

    5. *

      2. 由于默认一行行的读取excel,所以需要创建excel一行一行的回调监听器,参照{@link IndexOrNameDataListener}

    6. *

      3. 直接读即可

    7. */
    8. @Test
    9. public void indexOrNameRead() {
    10. String fileName = TestFileUtil.getPath() + "demo" + File.separator + "demo.xlsx";
    11. // 这里默认读取第一个sheet
    12. EasyExcel.read(fileName, DemoData.class, new DemoDataListener()).sheet().doRead();
    13. }

    2. 写Excel

    对象

    1. @Getter
    2. @Setter
    3. @EqualsAndHashCode
    4. public class DemoData {
    5. @ExcelProperty("字符串标题")
    6. private String string;
    7. @ExcelProperty("日期标题")
    8. private Date date;
    9. @ExcelProperty("数字标题")
    10. private Double doubleData;
    11. /**
    12. * 忽略这个字段
    13. */
    14. @ExcelIgnore
    15. private String ignore;
    16. }

    代码

    1. /**
    2. * 最简单的写
    3. *

    4. * 1. 创建excel对应的实体对象 参照{@link DemoData}
    5. *

    6. * 2. 直接写即可
    7. */
    8. @Test
    9. public void simpleWrite() {
    10. // 注意 simpleWrite在数据量不大的情况下可以使用(5000以内,具体也要看实际情况),数据量大参照 重复多次写入
    11. // 写法1 JDK8+
    12. // since: 3.0.0-beta1
    13. String fileName = TestFileUtil.getPath() + "simpleWrite" + System.currentTimeMillis() + ".xlsx";
    14. // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
    15. // 如果这里想使用03 则 传入excelType参数即可
    16. EasyExcel.write(fileName, DemoData.class)
    17. .sheet("模板")
    18. .doWrite(() -> {
    19. // 分页查询数据
    20. return data();
    21. });
    22. // 写法2
    23. fileName = TestFileUtil.getPath() + "simpleWrite" + System.currentTimeMillis() + ".xlsx";
    24. // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
    25. // 如果这里想使用03 则 传入excelType参数即可
    26. EasyExcel.write(fileName, DemoData.class).sheet("模板").doWrite(data());
    27. // 写法3
    28. fileName = TestFileUtil.getPath() + "simpleWrite" + System.currentTimeMillis() + ".xlsx";
    29. // 这里 需要指定写用哪个class去写
    30. try (ExcelWriter excelWriter = EasyExcel.write(fileName, DemoData.class).build()) {
    31. WriteSheet writeSheet = EasyExcel.writerSheet("模板").build();
    32. excelWriter.write(data(), writeSheet);
    33. }
    34. }

    七、其他说明

    3+版本的的easyexcel,使用poi 5+版本时,需要自己引入poi 5+版本的包,且手动排除:poi-ooxml-schemas,例如:

    1. com.alibaba
    2. easyexcel
    3. 3.1.0
    4. poi-ooxml-schemas
    5. org.apache.poi

    八、版本升级

    • 不建议跨大版本升级 尤其跨2个大版本

    • 大版本升级后建议相关内容重新测试下

    • 2+ 升级到 3+ ,下面3个地方不兼容:

    • 使用了自定义拦截器去修改样式的会出问题(不会编译报错)

    1. // 以前的写法
    2. @Override
    3. protected void setHeadCellStyle(Cell cell, Head head, Integer relativeRowIndex) {
    4. cell.setCellStyle(style);
    5. }
    6. // 现在的写法1
    7. // 这个写完也需要测试下 还是老代码 不管使用了什么拦截器 都可以这么写
    8. // 这个会导致格式化数据失效
    9. protected void setHeadCellStyle(CellWriteHandlerContext context) {
    10. cell.setCellStyle(style);
    11. // 这里要把 WriteCellData的样式清空, 不然后面还有一个拦截器 FillStyleCellWriteHandler 默认会将 WriteCellStyle 设置到
    12. // cell里面去 会导致自己设置的不一样
    13. context.getFirstCellData().setWriteCellStyle(null);
    14. }
    15. // 现在的写法2 推荐
    16. // 这个方案靠谱 以前用 poi的CellStyle 现在用 WriteCellStyle 入参基本都一致
    17. protected void setHeadCellStyle(CellWriteHandlerContext context) {
    18. // 第一个单元格
    19. // 只要不是头 一定会有数据 当然fill的情况 可能要context.getCellDataList() ,这个需要看模板,因为一个单元格会有多个 WriteCellData
    20. WriteCellData cellData = context.getFirstCellData();
    21. // 这里需要去cellData 获取样式
    22. // 很重要的一个原因是 WriteCellStyle 和 dataFormatData绑定的 简单的说 比如你加了 DateTimeFormat
    23. // ,已经将writeCellStyle里面的dataFormatData 改了 如果你自己new了一个WriteCellStyle,可能注解的样式就失效了
    24. // 然后 getOrCreateStyle 用于返回一个样式,如果为空,则创建一个后返回
    25. WriteCellStyle writeCellStyle = cellData.getOrCreateStyle();
    26. writeCellStyle.setFillForegroundColor(IndexedColors.RED.getIndex());
    27. // 这里需要指定 FillPatternType 为FillPatternType.SOLID_FOREGROUND
    28. writeCellStyle.setFillPatternType(FillPatternType.SOLID_FOREGROUND);
    29. }
    • 读的时候invoke里面抛出异常,不会再额外封装一层ExcelAnalysisException (不会编译报错)这个捕获异常的时候 不用再getCause了

    • 样式等注解涉及到 boolean or 一些枚举 值的 有变动,新增默认值(会编译报错,注解改就行)这个直接改了就行

    图片

  • 相关阅读:
    mysql语句
    【Oracle】Oracle系列之十六--数据库备份
    前端笔试练习题——JS7 无重复数组、JS8 数组排序
    Mybatis中的缓存相关简介说明
    记录一次数据库CPU被打满的排查过程
    ARM Cortex-M内核中系统堆栈
    【性能测试】Action.c(6): Error -26612
    【Spring】bean的基础配置
    Leetcode 381. Insert Delete GetRandom O(1) - Duplicates allowed (数据结构设计好题)
    美团笔试2022.8.6
  • 原文地址:https://blog.csdn.net/weixin_40381772/article/details/133299915