当前位置:   article > 正文

阿里开源EasyExcel 文件导入导出使用_阿里巴巴 easyexcel 导出

阿里巴巴 easyexcel 导出

一、概述

EasyExcel:
        阿里开源:关于Easyexcel | Easy Excel 官网
        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模式,在上层做了模型转换的封装,让使用者更加简单方便
节约内存


二、导出excel
        三种方式:
                方法1: 如果写到同一个sheet
                方法2: 如果写到不同的sheet 同一个对象
                方法3 如果写到不同的sheet 不同的对象

  1. /**
  2. * 重复多次写入
  3. * <p>
  4. * 1. 创建excel对应的实体对象 参照{@link ComplexHeadData}
  5. * <p>
  6. * 2. 使用{@link ExcelProperty}注解指定复杂的头
  7. * <p>
  8. * 3. 直接调用二次写入即可
  9. */
  10. @Test
  11. public void repeatedWrite() {
  12. // 方法1: 如果写到同一个sheet
  13. String fileName = TestFileUtil.getPath() + "repeatedWrite" + System.currentTimeMillis() + ".xlsx";
  14. // 这里 需要指定写用哪个class去写
  15. try (ExcelWriter excelWriter = EasyExcel.write(fileName, DemoData.class).build()) {
  16. // 这里注意 如果同一个sheet只要创建一次
  17. WriteSheet writeSheet = EasyExcel.writerSheet("模板").build();
  18. // 去调用写入,这里我调用了五次,实际使用时根据数据库分页的总的页数来
  19. for (int i = 0; i < 5; i++) {
  20. // 分页去数据库查询数据 这里可以去数据库查询每一页的数据
  21. List<DemoData> data = data();
  22. excelWriter.write(data, writeSheet);
  23. }
  24. }
  25. // 方法2: 如果写到不同的sheet 同一个对象
  26. fileName = TestFileUtil.getPath() + "repeatedWrite" + System.currentTimeMillis() + ".xlsx";
  27. // 这里 指定文件
  28. try (ExcelWriter excelWriter = EasyExcel.write(fileName, DemoData.class).build()) {
  29. // 去调用写入,这里我调用了五次,实际使用时根据数据库分页的总的页数来。这里最终会写到5个sheet里面
  30. for (int i = 0; i < 5; i++) {
  31. // 每次都要创建writeSheet 这里注意必须指定sheetNo 而且sheetName必须不一样
  32. WriteSheet writeSheet = EasyExcel.writerSheet(i, "模板" + i).build();
  33. // 分页去数据库查询数据 这里可以去数据库查询每一页的数据
  34. List<DemoData> data = data();
  35. excelWriter.write(data, writeSheet);
  36. }
  37. }
  38. // 方法3 如果写到不同的sheet 不同的对象
  39. fileName = TestFileUtil.getPath() + "repeatedWrite" + System.currentTimeMillis() + ".xlsx";
  40. // 这里 指定文件
  41. try (ExcelWriter excelWriter = EasyExcel.write(fileName).build()) {
  42. // 去调用写入,这里我调用了五次,实际使用时根据数据库分页的总的页数来。这里最终会写到5个sheet里面
  43. for (int i = 0; i < 5; i++) {
  44. // 每次都要创建writeSheet 这里注意必须指定sheetNo 而且sheetName必须不一样。这里注意DemoData.class 可以每次都变,我这里为了方便 所以用的同一个class
  45. // 实际上可以一直变
  46. WriteSheet writeSheet = EasyExcel.writerSheet(i, "模板" + i).head(DemoData.class).build();
  47. // 分页去数据库查询数据 这里可以去数据库查询每一页的数据
  48. List<DemoData> data = data();
  49. excelWriter.write(data, writeSheet);
  50. }
  51. }
  52. }



三、文件写入

        四种方式:

  1. @PostMapping("/add")
  2. public void simpleRead() {
  3. // 写法1:JDK8+ ,不用额外写一个DemoDataListener
  4. // since: 3.0.0-beta1
  5. String fileName = "demo" + File.separator + "demo.xlsx";
  6. // 这里默认每次会读取100条数据 然后返回过来 直接调用使用数据就行
  7. // 具体需要返回多少行可以在`PageReadListener`的构造函数设置
  8. EasyExcel.read(fileName, Partner.class, new PageReadListener<Partner>(dataList -> {
  9. for (Partner partner : dataList) {
  10. log.info("读取到一条数据{}", JSON.toJSONString(partner));
  11. }
  12. })).sheet().doRead();
  13. // 写法2:
  14. // 匿名内部类 不用额外写一个DemoDataListener
  15. fileName = "demo" + File.separator + "demo.xlsx";
  16. // 这里 需要指定读用哪个class去读,然后读取第一个sheet 文件流会自动关闭
  17. EasyExcel.read(fileName, Partner.class, new ReadListener<Partner>() {
  18. /**
  19. * 单次缓存的数据量
  20. */
  21. public static final int BATCH_COUNT = 100;
  22. /**
  23. *临时存储
  24. */
  25. private List<Partner> cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
  26. @Override
  27. public void invoke(Partner 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 ="demo" + File.separator + "demo.xlsx";
  50. // 这里 需要指定读用哪个class去读,然后读取第一个sheet 文件流会自动关闭
  51. EasyExcel.read(fileName, Partner.class, new DemoDataListener()).sheet().doRead();
  52. // 写法4
  53. fileName ="demo" + File.separator + "demo.xlsx";
  54. // 一个文件一个reader
  55. try (ExcelReader excelReader = EasyExcel.read(fileName, Partner.class, new DemoDataListener()).build()) {
  56. // 构建一个sheet 这里可以指定名字或者no
  57. ReadSheet readSheet = EasyExcel.readSheet(0).build();
  58. // 读取一个sheet
  59. excelReader.read(readSheet);
  60. }
  61. }



        第三种、第四种需要监听器:

  1. // 有个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传进去
  2. @Slf4j
  3. public class DemoDataListener implements ReadListener<Partner> {
  4. /**
  5. * 每隔5条存储数据库,实际使用中可以100条,然后清理list ,方便内存回收
  6. */
  7. private static final int BATCH_COUNT = 100;
  8. /**
  9. * 缓存的数据
  10. */
  11. private List<Partner> cachedDataList = ListUtils.newArrayListWithExpectedSize(BATCH_COUNT);
  12. /**
  13. * 假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。当然如果不用存储这个对象没用。
  14. */
  15. private Partner partner;
  16. public DemoDataListener() {
  17. // 这里是demo,所以随便new一个。实际使用如果到了spring,请使用下面的有参构造函数
  18. partner = new Partner();
  19. }
  20. /**
  21. * 如果使用了spring,请使用这个构造方法。每次创建Listener的时候需要把spring管理的类传进来
  22. *
  23. * @param partner
  24. */
  25. public DemoDataListener(Partner partner) {
  26. this.partner = partner;
  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(Partner 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. partner.save(cachedDataList);
  62. log.info("存储数据库成功!");
  63. }
  64. }



四、场景案列

1、文件写入

接口配置:

  1. @PostMapping("/import")
  2. public void importExcel(@RequestParam("file") MultipartFile file) throws IOException {
  3. log.info("处理导入请求");
  4. if (!file.isEmpty()) {
  5. // 获取上传文件的原始名称
  6. String originalFilename = file.getOriginalFilename();
  7. // 临时文件路径,可以根据实际情况调整,就是linux或者windows对应路径
  8. String tempPath = "D:\\ideaWorkSpace\\qdf_java\\" + originalFilename;
  9. // 将上传的Excel文件写入临时文件
  10. try (BufferedOutputStream outputStream = new BufferedOutputStream(new FileOutputStream(tempPath))) {
  11. byte[] bytes = file.getBytes();
  12. outputStream.write(bytes);
  13. }
  14. // 使用临时文件路径进行数据读取
  15. String fileName = tempPath;
  16. EasyExcel.read(fileName, Partner.class, new PageReadListener<Partner>(dataList -> {
  17. partnerService.patchAdd(dataList);
  18. for (Partner partner : dataList) {
  19. log.info("读取到一条数据{}", JSON.toJSONString(partner));
  20. }
  21. })).sheet().doRead();
  22. // 可选:在操作完成后删除临时文件(如果不打算保留)
  23. Files.deleteIfExists(Paths.get(fileName));
  24. } else {
  25. throw new RuntimeException("上传的文件为空");
  26. }
  27. }

image.png


2、文件写出

接口配置:

  1. @GetMapping("/export")
  2. public ResponseEntity<byte[]> repeatedWrite1() {
  3. log.info("处理导出请求");
  4. int pageSize = 500;
  5. ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
  6. try (ExcelWriter excelWriter = EasyExcel.write(byteArrayOutputStream, Partner.class).build()) {
  7. WriteSheet writeSheet = EasyExcel.writerSheet("模板").build(); //sheet名称
  8. int totalPageCount = partnerService.getTotalPageCount(pageSize); // 总页码
  9. System.out.println("totalPageCount:" + totalPageCount);
  10. for (int i = 0; i < totalPageCount; i++) {
  11. // 分页去数据库查询数据
  12. List<Partner> data = partnerService.findPartnersByPage(i, pageSize); // 查出每页数据
  13. log.info("第" + (i+1) + "页数据条数:" + data.size());
  14. // 将数据写入内存中的ExcelWriter
  15. excelWriter.write(data, writeSheet);
  16. }
  17. } catch (Exception e) {
  18. log.error("导出Excel时发生错误", e);
  19. return ResponseEntity.status(HttpStatus.INTERNAL_SERVER_ERROR).build();
  20. }
  21. // 将内存中的Workbook转换为字节数组
  22. byte[] excelBytes = byteArrayOutputStream.toByteArray();
  23. // 设置HTTP响应头,告知浏览器这是一个需要下载的文件
  24. HttpHeaders headers = new HttpHeaders();
  25. headers.add(HttpHeaders.CONTENT_DISPOSITION, "attachment; filename=repeatedWrite1_" + System.currentTimeMillis() + ".xlsx");
  26. headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);
  27. // 返回包含Excel内容的ResponseEntity,浏览器将下载文件
  28. return ResponseEntity.ok().headers(headers).body(excelBytes);
  29. }


浏览器直接响应弹框,以下为接口测试:

image.png

3、实体类注解配置

详细注解看官网:https://easyexcel.opensource.alibaba.com/docs/

  1. @Data
  2. public class Partner extends BaseDomain{
  3. @ExcelProperty("合伙人区域ID")
  4. private Long id;
  5. /**
  6. * 合伙人id
  7. */
  8. @ExcelProperty("管理员ID")
  9. private Long adminId;
  10. /**
  11. * 密钥
  12. */
  13. @ExcelIgnore // 忽略的字段
  14. private String secretKey;

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/天景科技苑/article/detail/998843
推荐阅读
相关标签
  

闽ICP备14008679号