java使用EasyExcel实现Sheet的复制与填充

 更新时间:2023年10月08日 08:22:37   作者:酸奶小肥阳  
EasyExcel是一个非常有用的工具,它提供了强大的模板填充功能,可以轻松解决各种业务需求,本文主要为大家介绍了如何使用EasyExcel实现模板Sheet复制与填充,需要的可以参考下

当在后端开发中处理Excel业务时,EasyExcel是一个非常有用的工具,它提供了强大的模板填充功能,可以轻松解决各种业务需求。在本文中,我将首先简要介绍EasyExcel模板填充功能的基本用法,然后提供一种根据业务需要执行多份复制与填充的实用方案。

EasyExcel模板填充功能的基本用法

使用EasyExcel的[模板填充功能]与[简单写]相比,可以diy工作表(sheet)的样式,让生成的excel更加美观且不必花费更多的开发时间。

使用简单写完成一个Excel报表

1.创建Java类,方便构建数据映射到Excel中;在类中使用注解配置单元格格式和标题

@ExcelProperty: 设置excel中标题名称

@ColumnWidth: 设置单元格的宽度

@Builder
@Data
@AllArgsConstructor
@NoArgsConstructor
public class CatDTO {
    @ExcelProperty(value = "姓名")
    private String name;
    @ExcelProperty(value = "品种")
    @ColumnWidth(15)
    private String breed;
    @ExcelProperty(value = "年龄")
    private Double age;
    @ExcelProperty(value = "描述")
    @ColumnWidth(30)
    private String desc;
}

2.使用EasyExcel提供的Write API,将数据写入Excel中。

@SpringBootTest
class DemoApplicationTests {
    @Test
    void writeTest1() {
        String fileName = "write-" + System.currentTimeMillis() + ".xlsx";
        // 构建数据
        List<CatDTO> catWriteData = List.of(
                CatDTO.builder().name("肥猫").breed("银渐层").age(1.2).desc("每天吃吃睡睡睡睡睡").build(),
                CatDTO.builder().name("虎妞").breed("德文卷毛猫").age(0.6).desc("活泼好动无敌可爱爱粘人").build()
        );
        EasyExcel.write(fileName, CatDTO.class).sheet().doWrite(catWriteData);
    }
}

生成效果

使用模板填充完成一个Excel报表

1.创建Java类

@Builder
@Data
@AllArgsConstructor
@NoArgsConstructor
public class CatFillDTO {
    private String name;
    private String breed;
    private Double age;
    private String desc;
}

2.构建模板,在数据填充所在列上,填上与之对应的实体类属性名;

3.使用EasyExcel提供的Fill API,获取模板文件,写入数据。

@SpringBootTest
class DemoApplicationTests {
    @Test
    void fillTest1() {
        String fileName = "fill-" + System.currentTimeMillis() + ".xlsx";
        // 构建数据
        List<CatFillDTO> catFillData = List.of(
                CatFillDTO.builder().name("肥猫").breed("银渐层").age(1.2).desc("每天吃吃睡睡睡睡睡").build(),
                CatFillDTO.builder().name("虎妞").breed("德文卷毛猫").age(0.6).desc("活泼好动无敌可爱爱粘人").build()
        );
        FileUtil.touch(fileName);
        // 加载模板
        InputStream resourceAsStream = ResourceUtil.getStreamSafe("temp/cat-template.xlsx");
        // 生成工作簿对象
        ExcelWriter excelWriter = EasyExcel.write(fileName).withTemplate(resourceAsStream).build();
        // 生成工作表对象
        WriteSheet writeSheet = EasyExcel.writerSheet().build();
        // 多组数据填充需要另起一行时 forceNewRow(true)
        FillConfig fillConfig = FillConfig.builder().forceNewRow(true).build();
        // 多组数据填充 List<实体类> 对象
        excelWriter.fill(catFillData, fillConfig, writeSheet);
        // 关闭excelWriter
        excelWriter.finish();
    }
}

4.生成效果

可以看出,使用模板填充可以省去在代码中配置字段名称、单元格大小等配置,在实现复杂的Excel样式或美观修饰上更有优势;

根据业务需要执行多份复制与填充的方案

在处理Excel时,你或许会遇到这样的问题。使用指定模板填充数据,数据根据某个特征进行划分,分别填进不同的工作表(sheet)中,这样,我们需要根据业务的需求, 来生成相应的模板表数量,再分别填充上数据。

实现思路

  • 使用POI API加载模板文件为Workbook对象
  • 使用POI中的cloneSheet API复制相应数量的模板工作表
  • 将复制好数量的工作簿文件转换成输入流,以便在EasyExcel withTemplate API中使用
  • 使用EasyExcel fill API填充数据。

(直接上业务代码,实现的功能为模板表1不执行复制,只进行填充,模板表2进行复制并填充)

@Service
@Slf4j
@RequiredArgsConstructor
public class SpeakerServiceImpl implements ISpeakerService {
   /**
    * 填充模板文件
    *
    * @param sheet1Data sheet1数据
    * @param promptMap  提示地图
    * @return {@code File}
    */
   public File fillTemplateFile(
           CustomSpeakerDemandExcelDTO sheet1Data, Map<String, List<CustomSpeakerPromptDTO>> promptMap) {
      try {
         // 构建模板
         byte[] asInputStream = this.buildTemplateFile(promptMap);
         // 创建临时导出文件
         File temporaryFile = Files.createTempFile("speaker_task_", ".xlsx").toFile();
         // easyExcel API 根据is作为模板填充数据,并写入临时文件temporaryFile
         ExcelWriter excelWriter = EasyExcel.write(temporaryFile)
                 .withTemplate(new ByteArrayInputStream(asInputStream))
                 .build();
         // 填充模板1的sheet
         WriteSheet writeSheet1 = EasyExcel.writerSheet(0).build();
         excelWriter.fill(sheet1Data, writeSheet1);
         // 填充模板2的多个sheet
         int sheetIndex = 1;
         for (String key : promptMap.keySet()) {
            WriteSheet writeSheet = EasyExcel.writerSheet(sheetIndex).build();
            // 写入数据到当前sheet
            List<CustomSpeakerPromptDTO> prompt = promptMap.get(key);
            excelWriter.fill(prompt, writeSheet);
            sheetIndex++;
         }
         excelWriter.finish();
         return temporaryFile;
      } catch (IOException e) {
         throw new RuntimeException(e);
      }
   }
   /**
    * 构建模板文件
    *
    * @param promptMap 提示地图
    * @return {@code byte[]}
    * @throws IOException ioexception
    */
   public byte[] buildTemplateFile(Map<String, List<CustomSpeakerPromptDTO>> promptMap) throws IOException {
      // 加载模板文件
      InputStream templateStream = ResourceUtil.getStreamSafe("temp/speaker-task-template.xlsx");
      Workbook workbook = WorkbookFactory.create(templateStream);
      // 设置模板sheet数量, 需要复制promptMap.size()-1 次工作表2
      boolean first = true;
      for (String key : promptMap.keySet()) {
         if (first) {
            // 第一组数据可以直接使用当前模板,不需要进行复制
            workbook.setSheetName(1, key);
         } else {
            // 复制模板到新工作表,并设置表名
            Sheet newSheet = workbook.cloneSheet(1);
            workbook.setSheetName(workbook.getSheetIndex(newSheet), key);
         }
         first = false;
      }
      ByteArrayOutputStream ops = new ByteArrayOutputStream();
      workbook.write(ops);
      byte[] byteArray = ops.toByteArray();
      // 原文件流后续已不使用,此处关闭
      templateStream.close();
      ops.close();
      return byteArray;
   }
}

以上就是java使用EasyExcel实现Sheet的复制与填充的详细内容,更多关于java EasyExcel的资料请关注脚本之家其它相关文章!

相关文章

  • MyBatis-Plus实现逻辑删除功能解析

    MyBatis-Plus实现逻辑删除功能解析

    这篇文章主要介绍了MyBatis-Plus实现逻辑删除功能解析,有时候并不需要真正的删除数据,而是想逻辑删除,方便数据恢复,MyBatis-Plus可以很方便的实现逻辑删除的功能,需要的朋友可以参考下
    2023-11-11
  • Java高并发下请求合并处理方式

    Java高并发下请求合并处理方式

    这篇文章主要介绍了Java高并发下请求合并处理方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-08-08
  • springboot打包jar中没有主清单属性问题

    springboot打包jar中没有主清单属性问题

    这篇文章主要介绍了springboot打包jar中没有主清单属性问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • Spring项目读取配置文件中文乱码的解决

    Spring项目读取配置文件中文乱码的解决

    这篇文章主要介绍了Spring项目读取配置文件中文乱码的解决方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-01-01
  • 使用springboot整合RateLimiter限流过程

    使用springboot整合RateLimiter限流过程

    这篇文章主要介绍了使用springboot整合RateLimiter限流过程,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-06-06
  • JDK序列化Bug难题解决示例详解

    JDK序列化Bug难题解决示例详解

    这篇文章主要为大家介绍了JDK序列化Bug难题解决示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • Spring @Cacheable读取配置常量方式

    Spring @Cacheable读取配置常量方式

    这篇文章主要介绍了Spring @Cacheable读取配置常量方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • Java后端用EL表达式改进JSP

    Java后端用EL表达式改进JSP

    EL 全名为Expression Language,EL的语法很简单,它最大的特点就是使用上很方便,本文带你用EL表达式改进JSP,感兴趣的朋友来看看吧
    2022-02-02
  • 深入了解java 8的函数式编程

    深入了解java 8的函数式编程

    函数式编程并不是Java新提出的概念,其与指令编程相比,强调函数的计算比指令的计算更重要;与过程化编程相比,其中函数的计算可以随时调用。下面我们来详细了解一下吧
    2019-06-06
  • ELK搭建线上日志收集系统

    ELK搭建线上日志收集系统

    ELK日志收集系统进阶使用,本文主要讲解如何打造一个线上环境真实可用的日志收集系统,有了它,你就可以和去服务器上捞日志说再见了
    2022-07-07

最新评论