• Springboot+Easyexcel将数据写入模板文件并导出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模式,在上层做了模型转换的封装,让使用者更加简单方便。

    本文使用easyexcel对excel文件进行操作,来实现数据以excel形式导出的功能。

    需求背景:

    针对用户选择的数据,利用现有的excel模板,实现批量导出的功能。比如CSDN也有类似的批量导出数据功能,这里需要导出时,按照模板文件的字段导出。

    现有模板字段:

    实现方式:springboot+easyexcel

    一、导入依赖

    1. <dependency>
    2. <groupId>org.projectlombokgroupId>
    3. <artifactId>lombokartifactId>
    4. dependency>
    5. <dependency>
    6. <groupId>com.alibabagroupId>
    7. <artifactId>easyexcelartifactId>
    8. <version>3.0.5version>
    9. dependency>

    二、根据excel表头创建对应的实体类Pojo

    ExcelTitile实体类

    1. import com.alibaba.excel.annotation.ExcelProperty;
    2. import lombok.Data;
    3. @Data
    4. public class ExcelTitle {
    5. @ExcelProperty(value="事件名称", index=0)
    6. private String eventName;
    7. @ExcelProperty(value="需求负责人", index=1)
    8. private String prdManager;
    9. @ExcelProperty(value="技术负责人", index=2)
    10. private String techManager;
    11. @ExcelProperty(value="文档链接", index=3)
    12. private String prdDocs;
    13. @ExcelProperty(value="数据链接", index=4)
    14. private String statsDocs;
    15. @ExcelProperty(value="统计口径", index=5)
    16. private String reportCaliber;

    这里采用了@ExcelProperty的注解,其中value表示列名,index表示列名的索引值。@Data 注解的主要作用是提高代码的简洁,使用这个注解可以省去代码中大量的get()、 set()、 toString()等方法;

    三、Controller类接收请求

    1. @RequestMapping(value = "/bulkOutput", method = RequestMethod.GET)
    2. public ResultBean bulkOutput(HttpServletResponse response) {
    3. // 重要! 设置返回格式是excel形式
    4. response.setContentType("application/vnd.openxmlformats-officedocument.spreadsheetml.sheet");
    5. // 设置编码格式
    6. response.setCharacterEncoding("utf-8");
    7. // 设置URLEncoder.encode 防止中文乱码
    8. String fileName = null;
    9. try {
    10. fileName = URLEncoder.encode("数据批量导出", "UTF-8").replaceAll("\\+", "%20");
    11. // 设置响应头
    12. response.setHeader("Content-disposition", "attachment;filename*=utf-8''" + fileName + ".xlsx");
    13. List bulkOutputData = wildEventService.getBulkOutputData();
    14. // 模板文件保存在springboot项目的resources/static下
    15. Resource resource = new ClassPathResource("static/数据批量导出模板.xlsx");
    16. ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream())
    17. .withTemplate(resource.getInputStream()) // 利用模板的输出流
    18. .build();
    19. // 写入模板文件的第一个sheet 索引0
    20. WriteSheet writeSheet = EasyExcel.writerSheet(0).build();
    21. // 将数据写入到模板文件的对应sheet中
    22. excelWriter.write(bulkOutputData, writeSheet);
    23. excelWriter.finish();
    24. } catch (UnsupportedEncodingException e) {
    25. return ResultBean.errorService(e.getMessage());
    26. } catch (IOException e) {
    27. return ResultBean.errorService(e.getMessage());
    28. }
    29. return ResultBean.success("数据导出成功!");
    30. }

    上述代码中,首先对response进行了设置,设置了返回类型,响应头,以及导出下载时的文件名称。接下来,利用Resource resource = new ClassPathResource("static/数据批量导出模板.xlsx"); 读取项目下的模板文件,并调用easyexcel的写入方法。这里write(response.getOutputStream())表示写入response的输出流,即将文件返回给客户端进行下载withTemplate(resource.getInputStream())表示读取模板文件进行写入。最后调用 WriteSheet writeSheet = EasyExcel.writerSheet(0).build();将苏剧写入模板文件的第一个sheet中(索引从0开始)。

    四、Service层获取待写入数据

    这里为了简易期间,构造一些数据如下

    1. import com.example.demo.Pojo.ExcelTitle;
    2. import org.springframework.stereotype.Service;
    3. import java.util.ArrayList;
    4. import java.util.List;
    5. @Service
    6. public class BulkOutputService {
    7. public List getOutputData()
    8. {
    9. List resultList = new ArrayList<>();
    10. // 第一行数据
    11. ExcelTitle t1 = new ExcelTitle();
    12. t1.setEventName("测试数据1");
    13. t1.setTechManager("张三");
    14. t1.setPrdManager("张三");
    15. t1.setPrdDocs("http://prdDocs.com");
    16. t1.setStatsDocs("http://statsDocs.com");
    17. t1.setReportCaliber("reportCaliber");
    18. // 第二行数据
    19. ExcelTitle t2 = new ExcelTitle();
    20. t2.setEventName("测试数据2");
    21. t2.setTechManager("李四");
    22. t2.setPrdManager("李四");
    23. t2.setPrdDocs("http://prdDocs.com");
    24. t2.setStatsDocs("http://statsDocs.com");
    25. t2.setReportCaliber("reportCaliber");
    26. resultList.add(t1);
    27. resultList.add(t2);
    28. return resultList;
    29. }
    30. }

    这里构造了两行简易数据,实际场景中只需要根据自己的数据做替换就可以了。

    有了上述步骤,就可以接下来执行代码查看导出效果了。

    效果展示

    启动springboot程序,在浏览器中输入请求: http://localhost:8080/bulkOutput, 可以发现返回的excel文件被浏览器下载,打开后内容如下:

    可以看到,数据根据模板格式写入了文件,并以excel的形式导出。初步完成了我们预定的场景。

    总结

    本次实现的功能是利用现有模板,将数据批量导出成excel,借助于easyexcel来实现操作excel的功能。最重要的功能模块是Controller里的内容。

    需要注意

    1. 需要加上response的响应类型和响应头来使得返回请求返回excel文件

    2. 写到Web流时,这里的ContentType和CharacterEncoding不要乱码,否则很容易乱码或者文件损坏

    3. 使用EasyExcel.withTemplate引入模板的输入流

  • 相关阅读:
    stm32工程中的DebugConfig、Listings和Objects 3个文件夹
    (附源码)计算机毕业设计SSM基于框架的在线问答平台
    Golang项目实战(三)
    Java—多态
    PHP中=>和->有什么区别?
    Charles抓取接口报文并修改各种参数信息调试
    c 语言基础:L1-045 宇宙无敌大招呼
    灵活使用ssh、dsh和pssh高效管理大量计算机
    PyTorch实战 | 文本情感分类任务 | LSTM与LSTM+Attention
    第二十一节——路由初识
  • 原文地址:https://blog.csdn.net/LBWNB_Java/article/details/126133911