帮助文档:https://www.yuque.com/easyexcel/doc/easyexcel
POI操作Excel导入maven依赖
<!--Java 万物皆对象--> <!--xls(03)--> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>3.9</version> </dependency> <!--xlsx(07)--> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>3.9</version> </dependency> <!--日期格式化工具--> <dependency> <groupId>joda-time</groupId> <artifactId>joda-time</artifactId> <version>2.10.1</version> </dependency> <!--test--> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> </dependency> <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-simple</artifactId> <version>1.7.25</version> </dependency>代码实现
private static final String path = "C:\\Program Files\\JetBrains\\workspace\\poi-easyexcel\\"; /** * 03 xls * 优点:过程中写入缓存,不操作磁盘,最后一次性写入磁盘,速度快 * 缺点:最多65536行,超出异常 * @throws IOException */ @Test public void testWrite03() throws IOException { // 1.创建一个工作簿(bu) Workbook workbook = new HSSFWorkbook(); // 2.创建一个工作表 Sheet sheet = workbook.createSheet("观众统计表"); // 3.创建一个行 Row row1 = sheet.createRow(0); // 4.创建一个单元格 Cell cell1 = row1.createCell(0); cell1.setCellValue("今日新增观众"); Cell cell2 = row1.createCell(1); cell2.setCellValue(666); // 第二行 Row row2 = sheet.createRow(1); Cell cell21 = row2.createCell(0); cell21.setCellValue("统计时间"); Cell cell22 = row2.createCell(1); String date = new DateTime().toString("yyyy-MM-dd HH:mm:ss"); cell22.setCellValue(date); // 生成一张表(IO流)03 版本就是使用 xls结尾! FileOutputStream fileOutputStream = new FileOutputStream(path + "观众统计表03.xls"); workbook.write(fileOutputStream); // 关闭流 fileOutputStream.close(); } /** * 07 xlsx * 优点:可以写入较大的数据,如20万条 * 缺点:写入数据速度非常慢,非常耗内存,也会发生内存溢出,如100万条 * @throws IOException */ @Test public void testWrite07() throws IOException { // 1.创建一个工作簿(bu) Workbook workbook = new XSSFWorkbook(); // 2.创建一个工作表 Sheet sheet = workbook.createSheet("观众统计表"); // 3.创建一个行 Row row1 = sheet.createRow(0); // 4.创建一个单元格 Cell cell1 = row1.createCell(0); cell1.setCellValue("今日新增观众"); Cell cell2 = row1.createCell(1); cell2.setCellValue(666); // 第二行 Row row2 = sheet.createRow(1); Cell cell21 = row2.createCell(0); cell21.setCellValue("统计时间"); Cell cell22 = row2.createCell(1); String date = new DateTime().toString("yyyy-MM-dd HH:mm:ss"); cell22.setCellValue(date); // 生成一张表(IO流)07 版本就是使用 xlsx结尾! FileOutputStream fileOutputStream = new FileOutputStream(path + "观众统计表07.xlsx"); workbook.write(fileOutputStream); // 关闭流 fileOutputStream.close(); } /** * 03 HSSFWorkbook xls * 07 XSSFWorkbook xlxs * 07 SXSSFWorkbook xlsx增强版 * * @throws FileNotFoundException */ @Test public void testCellType() throws IOException { // 获取文件流 FileInputStream fileInputStream = new FileInputStream(path + "会员消费商品明细表.xls"); // 1.创建一个工作簿 Workbook workbook = new HSSFWorkbook(fileInputStream); // 获取标题内容 Sheet sheet = workbook.getSheetAt(0); // 获取标题内容 Row rowTitle = sheet.getRow(0); if (rowTitle != null) { int physicalNumberOfCells = rowTitle.getPhysicalNumberOfCells(); // 多少列 for (int cellNum = 0; cellNum < physicalNumberOfCells; cellNum++) { Cell cell = rowTitle.getCell(cellNum); if (cell != null) { int cellType = cell.getCellType(); String stringCellValue = cell.getStringCellValue(); log.info(stringCellValue + "|"); } } } // 获取表中内容 int physicalNumberOfRows = sheet.getPhysicalNumberOfRows(); for (int rowNum = 1; rowNum < physicalNumberOfRows; rowNum++) { Row rowData = sheet.getRow(rowNum); if (rowData != null) { // 读取列 int physicalNumberOfCells = rowData.getPhysicalNumberOfCells(); for (int cellNum = 0; cellNum < physicalNumberOfCells; cellNum++) { log.info("[" + (rowNum + 1) + "-" + (cellNum + 1) + "]"); Cell cell = rowData.getCell(cellNum); // 匹配的数据类型 if (cell != null) { int cellType = cell.getCellType(); String cellValue = ""; switch (cellType) { case HSSFCell .CELL_TYPE_STRING: // 字符串 cellValue = cell.getStringCellValue(); break; case HSSFCell. CELL_TYPE_BOOLEAN: // 布尔 cellValue = String.valueOf(cell.getBooleanCellValue()); break; case HSSFCell. CELL_TYPE_BLANK: // 空 break; case HSSFCell.CELL_TYPE_NUMERIC: //数字(日期、普通数字) if (HSSFDateUtil.isCellDateFormatted(cell)) { // 日期 Date dateCellValue = cell.getDateCellValue(); cellValue = new DateTime(dateCellValue).toString("yyyy-MM-dd"); } else { // 普通数字 cell.setCellType(HSSFCell.CELL_TYPE_STRING); cellValue = cell.toString(); } break; case HSSFCell.CELL_TYPE_ERROR: log.info(cell.toString()); break; } log.info(cellValue); } } } } fileInputStream.close(); } /** * Excel计算公式 * @throws IOException */ @Test public void testFormula() throws IOException { FileInputStream fileInputStream = new FileInputStream(path + "计算公式.xls"); Workbook workbook = new HSSFWorkbook(fileInputStream); Sheet sheet = workbook.getSheetAt(0); Row row = sheet.getRow(4); Cell cell = row.getCell(0); // 拿到计算公式 excel FormulaEvaluator formulaEvaluator = new HSSFFormulaEvaluator((HSSFWorkbook)workbook); // 输出单元格内容 int cellType = cell.getCellType(); switch (cellType) { case Cell.CELL_TYPE_FORMULA: // 公式 String cellFormula = cell.getCellFormula(); log.info(cellFormula); // 计算 CellValue evaluate = formulaEvaluator.evaluate(cell); String cellValue = evaluate.formatAsString(); log.info(cellValue); break; } }2.EasyExcel读写
<dependency> <groupId>com.alibaba</groupId> <artifactId>easyexcel</artifactId> <version>2.2.6</version> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>1.18.12</version> </dependency> <!--日期格式化工具--> <dependency> <groupId>joda-time</groupId> <artifactId>joda-time</artifactId> <version>2.10.1</version> </dependency> <!--test--> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> </dependency> <dependency> <groupId>org.slf4j</groupId> <artifactId>slf4j-simple</artifactId> <version>1.7.25</version> </dependency> <dependency> <groupId>com.alibaba</groupId> <artifactId>fastjson</artifactId> <version>1.2.73</version> </dependency> package com.sunreal; import com.alibaba.excel.EasyExcel; import org.junit.Test; import java.util.ArrayList; import java.util.Date; import java.util.List; /** * @author lkz * @description * @date 2020/10/5 17:05 */ public class EasyTest { private static final String path = "C:\\Program Files\\JetBrains\\workspace\\poi-easyexcel\\"; private List<DemoData> data() { List<DemoData> list = new ArrayList<DemoData>(); DemoData demoData = null; for (int i = 0; i < 10; i++) { demoData = new DemoData(); demoData.setString("字符串" + i); demoData.setDate(new Date()); demoData.setDoubleData(0.56); list.add(demoData); } return list; } /** * 根据list 写入Exel */ @Test public void simpleWrite() { String fileName = path + "EasyExcelTest.xlsx"; // 写法1 // write(fileName,格式类) // sheet(表名) // doWrite(数据) EasyExcel.write(fileName, DemoData.class).sheet("模板").doWrite(data()); // 写法2 // ExcelWriter excelWriter = EasyExcel.write(fileName, DemoData.class).build(); // WriteSheet writeSheet = EasyExcel.writerSheet("模板").build(); // excelWriter.write(data(), writeSheet); // // 千万别忘记finish,会帮忙关闭流 // excelWriter.finish(); } /** * 读取Exel */ @Test public void simpleRead() { // 有个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new,然后里面用到spring可以构造方法传递 String fileName = path + "EasyExcelTest.xlsx"; EasyExcel.read(fileName, DemoData.class, new DemoDataListener()).sheet().doRead(); } } package com.sunreal; import com.alibaba.excel.annotation.ExcelIgnore; import com.alibaba.excel.annotation.ExcelProperty; import lombok.Data; import java.util.Date; /** * @author lkz * @description * @date 2020/10/5 16:48 */ @Data public class DemoData { @ExcelProperty("字符串标题") private String string; @ExcelProperty("日期标题") private Date date; @ExcelProperty("数字标题") private Double doubleData; /** * 忽略这个字段注解@ExcelIgnore */ @ExcelIgnore private String ignore; } package com.sunreal; import netscape.javascript.JSObject; import java.util.List; /** * @author lkz * @description 假设这个是你的DAO存储。 * 当然还要这个类让Spring管理,当然你不需要存储,也不需要这个类。 * @date 2020/10/5 17:11 */ public class DemoDAO { public void save(List<DemoData> list) { // 持久化操作 // 如果是mybatis,尽量别直接调用多次insert,自己写一个mapper里面新增一个方法batchInsert,所有数据一次性插入 } } package com.sunreal; import com.alibaba.excel.context.AnalysisContext; import com.alibaba.excel.event.AnalysisEventListener; import com.alibaba.fastjson.JSONObject; import org.slf4j.Logger; import org.slf4j.LoggerFactory; import java.util.ArrayList; import java.util.List; /** * @author lkz * @description 有一个很重要的点 DemoDataListener 不能被spring管理,要每次读取excel都要new, * 然后里面用的spring可以构造方法传进去 * @date 2020/10/5 17:14 */ public class DemoDataListener extends AnalysisEventListener<DemoData> { private static final Logger log = LoggerFactory.getLogger(DemoDataListener.class); /** * 每隔5条存储数据库,实际使用中可以3000条,然后清理list,方便内存回收 */ private static final int BATCH_COUNT = 5; List<DemoData> list = new ArrayList<DemoData>(); /** * 假设这个是一个DAO,当然有业务逻辑这个也可以是一个service。当然如果不用存储这个对象没用。 */ private DemoDAO demoDAO; public DemoDataListener() { demoDAO = new DemoDAO(); } /** * 如果使用了spring,请使用这个构造方法。每次创建Listener的时候需要把spring管理的类传进来 * * @param demoDAO */ public DemoDataListener(DemoDAO demoDAO) { this.demoDAO = demoDAO; } /** * 这个每一条数据解析都会调用 * * @param demoData 类型 * @param analysisContext 分析上下文 */ public void invoke(DemoData demoData, AnalysisContext analysisContext) { log.info(JSONObject.toJSONString(demoData)); list.add(demoData); // 达到BATCH_COUNT了,需要去存储一次数据库,防止数据几万条数据在内存,容易OOM if (list.size() >= BATCH_COUNT) { saveData(); // 存储完成清理 list list.clear(); } } /** * 所有数据解析完成了 都会来调用 * @param analysisContext */ public void doAfterAllAnalysed(AnalysisContext analysisContext) { // 这里也要保存数据,确保最后遗留的数据也存储到数据库 saveData(); // 持久化 log.info("所有数据解析完成!"); } /** * 加上存储数据库 */ private void saveData() { demoDAO.save(list); } }