EasyExcel

打印 上一主题 下一主题

主题 942|帖子 942|积分 2826

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

导入依赖
  1. <dependency>
  2.         <groupId>com.alibaba</groupId>
  3.     <artifactId>easyexcel</artifactId>
  4.     <version>3.1.1</version>
  5. </dependency>
复制代码
写 Excel

1.以实体类映射想要写入 Excel 的数据
  1. @Data
  2. @Builder
  3. @NoArgsConstructor
  4. @AllArgsConstructor
  5. public class ExcelEntity {
  6.     @ExcelProperty("字符串标题")
  7.     private String string;
  8.     @ExcelProperty("日期标题")
  9.     private Date date;
  10.     @ExcelProperty("数字标题")
  11.     private Double doubleData;
  12.     @ExcelIgnore
  13.     private String ignore;
  14. }
复制代码
注解说明

  • ExcelProperty:定义排版

    • value:字符串标题
    • index:指定数据所在列,从0起算
      1. @ExcelProperty(value = "字符串标题", index = 0)
      复制代码
    • 复杂头写入
      1. @ExcelProperty({"主标题", "字符串标题"})
      复制代码

  • DateTimeFormat:定义时间格式
    1. @DateTimeFormat("yyyy年MM月dd日HH时mm分ss秒")
    2. @ExcelProperty("日期标题")
    3. private Date date;
    复制代码
  • NumberFormat:定义数字格式
    1. @NumberFormat("#.##%")        // 用百分比表示
    2. @ExcelProperty(value = "数字标题")
    3. private Double doubleData;
    复制代码
  • ColumnWidth:列宽
  • ContentRowHeight:行高
  • HeadRowHeight:表头行高(写在类名上)
  • ContentLoopMerge:合并单元格
    1. // 这一列 每隔2行 合并单元格
    2. @ContentLoopMerge(eachRow = 2)
    3. @ExcelProperty("字符串标题")
    4. private String string;
    复制代码
  • ExcelIgnore:忽略这个字段
2.将数据存储到实体类中
  1.     private List<DemoData> data() {
  2.         List<DemoData> list = ListUtils.newArrayList();
  3.         for (int i = 0; i < 10; i++) {
  4.             DemoData data = new DemoData();
  5.             data.setString("字符串" + i);
  6.             data.setDate(new Date());
  7.             data.setDoubleData(0.56);
  8.             list.add(data);
  9.         }
  10.         return list;
  11.     }
复制代码
3.数据写入 Excel

方法1:在数据量不大的情况下可以使用(5000以内,具体也要看实际情况),数据量大参照 重复多次写入
  1.     @Test
  2.     public void simpleWrite() {
  3.         // 创建一个文件的绝对路径
  4.         String fileName = "D:\\demo1.xlsx";
  5.         // write(): 参数一:文件路径    参数二:要写的数据对应的实体字节码
  6.         // sheet(): excel表格名称
  7.         // doWrite(): 要写的数据,是一个集合
  8.         EasyExcel.write(fileName, ExcelEntity.class).sheet("模板").doWrite(() -> {
  9.             // 分页查询数据
  10.             return data();
  11.         });
  12.     }
复制代码
方法2:
  1.     @Test
  2.     public void simpleWrite() {
  3.         
  4.         String fileName = "D:\\demo1.xlsx";
  5.         // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
  6.         // 如果这里想使用03 则 传入excelType参数即可
  7.         EasyExcel.write(fileName, ExcelEntity.class).sheet("模板").doWrite(data());
  8.     }
复制代码
方法3:
  1.     @Test
  2.     public void simpleWrite3() {
  3.         String fileName = "D:\\demo1.xlsx";
  4.         // 这里 需要指定写用哪个class去写
  5.         try (ExcelWriter excelWriter = EasyExcel.write(fileName, ExcelEntity.class).build()) {
  6.             WriteSheet writeSheet = EasyExcel.writerSheet("模板").build();
  7.             excelWriter.write(data(), writeSheet);
  8.         }
  9.     }
复制代码
4.忽略或指定导出

忽略
  1.         // 根据用户传入字段 假设我们要忽略 date
  2.         Set<String> excludeColumnFiledNames = new HashSet<String>();
  3.         excludeColumnFiledNames.add("date");
  4.         // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
  5.         EasyExcel.write(fileName, ExcelEntity.class).excludeColumnFiledNames(excludeColumnFiledNames).sheet("模板")
  6.                 .doWrite(data());
复制代码
指定导出
  1.         // 根据用户传入字段 假设我们只要导出 date
  2.         Set<String> includeColumnFiledNames = new HashSet<String>();
  3.         includeColumnFiledNames.add("date");
  4.         // 这里 需要指定写用哪个class去写,然后写到第一个sheet,名字为模板 然后文件流会自动关闭
  5.         EasyExcel.write(fileName, ExcelEntity.class).includeColumnFiledNames(includeColumnFiledNames).sheet("模板")
  6.                 .doWrite(data());
复制代码
5.在同一 sheet 续写
  1.     @Test
  2.     public void repeatedWrite() {
  3.         
  4.         String fileName = "D:\\demo1.xlsx";
  5.         // 这里 需要指定写用哪个class去写
  6.         try (ExcelWriter excelWriter = EasyExcel.write(fileName, ExcelEntity.class).build()) {
  7.             // 这里注意 如果同一个sheet只要创建一次
  8.             WriteSheet writeSheet = EasyExcel.writerSheet("模板").build();
  9.             // 去调用写入,这里我调用了五次,实际使用时根据数据库分页的总的页数来
  10.             for (int i = 0; i < 5; i++) {
  11.                 // 分页去数据库查询数据 这里可以去数据库查询每一页的数据
  12.                 List<ExcelEntity> data = data();
  13.                 excelWriter.write(data, writeSheet);
  14.             }
  15.         }
  16.     }
复制代码
6.同一个对象写到不同的sheet
  1.     @Test
  2.     public void repeatedWrite() {
  3.         
  4.         String fileName = "D:\\demo1.xlsx";
  5.         // 这里 指定文件
  6.         try (ExcelWriter excelWriter = EasyExcel.write(fileName, ExcelEntity.class).build()) {
  7.             // 去调用写入,这里我调用了五次,实际使用时根据数据库分页的总的页数来。这里最终会写到5个sheet里面
  8.             for (int i = 0; i < 5; i++) {
  9.                 // 每次都要创建writeSheet 这里注意必须指定sheetNo 而且sheetName必须不一样
  10.                 WriteSheet writeSheet = EasyExcel.writerSheet(i, "模板" + i).build();
  11.                 // 分页去数据库查询数据 这里可以去数据库查询每一页的数据
  12.                 List<ExcelEntity> data = data();
  13.                 excelWriter.write(data, writeSheet);
  14.             }
  15.         }
  16.     }
复制代码
7.不同的对象写到不同的sheet
  1.     @Test
  2.     public void repeatedWrite() {
  3.         String fileName = "D:\\demo1.xlsx";
  4.         // 这里 指定文件
  5.         try (ExcelWriter excelWriter = EasyExcel.write(fileName).build()) {
  6.             // 去调用写入,这里我调用了五次,实际使用时根据数据库分页的总的页数来。这里最终会写到5个sheet里面
  7.             for (int i = 0; i < 5; i++) {
  8.                 // 每次都要创建writeSheet 这里注意必须指定sheetNo 而且sheetName必须不一样。这里注意DemoData.class 可以每次都变,我这里为了方便 所以用的同一个class
  9.                 // 实际上可以一直变
  10.                 WriteSheet writeSheet = EasyExcel.writerSheet(i, "模板" + i).head(ExcelEntity.class).build();
  11.                 // 分页去数据库查询数据 这里可以去数据库查询每一页的数据
  12.                 List<ExcelEntity> data = data();
  13.                 excelWriter.write(data, writeSheet);
  14.             }
  15.         }
  16.     }
复制代码
读 Excel

1.以实体类映射想要读取的 Excel 数据
  1. @Data
  2. @Builder
  3. @NoArgsConstructor
  4. @AllArgsConstructor
  5. public class ExcelEntity {
  6.     private String string;
  7.     private Date date;
  8.     private Double doubleData;
  9. }
复制代码
2.持久层
  1. /**
  2. * 假设这个是你的DAO存储。当然还要这个类让spring管理,当然你不用需要存储,也不需要这个类。
  3. **/
  4. public class ExcelDAO {
  5.     public void save(List<ExcelEntity> list) {
  6.         // 如果是mybatis,尽量别直接调用多次insert,自己写一个mapper里面新增一个方法batchInsert,所有数据一次性插入
  7.     }
  8. }
复制代码
3.监听器
  1. // 有个很重要的点 ExcelEntityListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去
  2. public class ExcelEntityListener implements ReadListener<ExcelEntity> {
  3.     /**
  4.      * 每隔5条存储数据库,实际使用中可以100条,然后清理list ,方便内存回收
  5.      */
  6.     private static final int BATCH_COUNT = 100;
  7.     /**
  8.      * 缓存的数据
  9.      */
  10.     private List<ExcelEntity> cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
  11.     /**
  12.      * 假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。当然如果不用存储这个对象没用。
  13.      */
  14.     private ExcelDAO excelDAO;
  15.     public ExcelEntityListener() {
  16.         // 这里是demo,所以随便new一个。实际使用如果到了spring,请使用下面的有参构造函数
  17.         excelDAO = new ExcelDAO();
  18.     }
  19.     /**
  20.      * 如果使用了spring,请使用这个构造方法。每次创建Listener的时候需要把spring管理的类传进来
  21.      *
  22.      * @param excelDAO
  23.      */
  24.     public ExcelEntityListener(ExcelDAO excelDAO) {
  25.         this.excelDAO = excelDAO;
  26.     }
  27.     /**
  28.      * 这个每一条数据解析都会来调用
  29.      *
  30.      * @param data    one row value. Is is same as {@link AnalysisContext#readRowHolder()}
  31.      * @param context
  32.      */
  33.     @Override
  34.     public void invoke(ExcelEntity data, AnalysisContext context) {
  35.         cachedDataList.add(data);
  36.         // 达到BATCH_COUNT了,需要去存储一次数据库,防止数据几万条数据在内存,容易OOM
  37.         if (cachedDataList.size() >= BATCH_COUNT) {
  38.             saveData();
  39.             // 存储完成清理 list
  40.             cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
  41.         }
  42.     }
  43.     /**
  44.      * 所有数据解析完成了 都会来调用
  45.      *
  46.      * @param context
  47.      */
  48.     @Override
  49.     public void doAfterAllAnalysed(AnalysisContext context) {
  50.         // 这里也要保存数据,确保最后遗留的数据也存储到数据库
  51.         saveData();
  52.     }
  53.     /**
  54.      * 加上存储数据库
  55.      */
  56.     private void saveData() {
  57.         excelDAO.save(cachedDataList);
  58.     }
  59. }
复制代码
4.读取 Excel 数据
  1.     @Test
  2.     public void simpleRead() {
  3.         // 写法1:JDK8+ ,不用额外写一个DemoDataListener
  4.         // since: 3.0.0-beta1
  5.         String fileName = "D:\\damo1.xlsx";
  6.         // 这里默认每次会读取100条数据 然后返回过来 直接调用使用数据就行
  7.         // 具体需要返回多少行可以在`PageReadListener`的构造函数设置
  8.         EasyExcel.read(fileName, ExcelEntity.class, new PageReadListener<ExcelEntity>(dataList -> {
  9.             for (ExcelEntity demoData : dataList) {
  10.                 log.info("读取到一条数据{}", JSON.toJSONString(demoData));
  11.             }
  12.         })).sheet().doRead();
  13.         // 写法2:
  14.         // 匿名内部类 不用额外写一个DemoDataListener
  15.         fileName = "D:\\damo1.xlsx";
  16.         // 这里 需要指定读用哪个class去读,然后读取第一个sheet 文件流会自动关闭
  17.         EasyExcel.read(fileName, ExcelEntity.class, new ReadListener<ExcelEntity>() {
  18.             /**
  19.              * 单次缓存的数据量
  20.              */
  21.             public static final int BATCH_COUNT = 100;
  22.             /**
  23.              *临时存储
  24.              */
  25.             private List<ExcelEntity> cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
  26.             @Override
  27.             public void invoke(ExcelEntity data, AnalysisContext context) {
  28.                 cachedDataList.add(data);
  29.                 if (cachedDataList.size() >= BATCH_COUNT) {
  30.                     saveData();
  31.                     // 存储完成清理 list
  32.                     cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
  33.                 }
  34.             }
  35.             @Override
  36.             public void doAfterAllAnalysed(AnalysisContext context) {
  37.                 saveData();
  38.             }
  39.             /**
  40.              * 加上存储数据库
  41.              */
  42.             private void saveData() {
  43.                 log.info("{}条数据,开始存储数据库!", cachedDataList.size());
  44.                 log.info("存储数据库成功!");
  45.             }
  46.         }).sheet().doRead();
  47.         // 有个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去
  48.         // 写法3:
  49.         fileName = "D:\\damo1.xlsx";
  50.         // 这里 需要指定读用哪个class去读,然后读取第一个sheet 文件流会自动关闭
  51.         EasyExcel.read(fileName, ExcelEntity.class, new ExcelEntityListener()).sheet().doRead();
  52.         // 写法4
  53.         fileName = "D:\\damo1.xlsx";
  54.         // 一个文件一个reader
  55.         try (ExcelReader excelReader = EasyExcel.read(fileName, ExcelEntity.class, new ExcelEntityListener()).build()) {
  56.             // 构建一个sheet 这里可以指定名字或者no
  57.             ReadSheet readSheet = EasyExcel.readSheet(0).build();
  58.             // 读取一个sheet
  59.             excelReader.read(readSheet);
  60.         }
  61.     }
  62. }
复制代码
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

嚴華

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表