JAVA实现easyExcel批量导入

注解类型描述
ExcelProperty导入指定当前字段对应excel中的那一列。可以根据名字或者Index去匹配。当然也可以不写,默认第一个字段就是index=0,以此类推。千万注意,要么全部不写,要么全部用index,要么全部用名字去匹配。千万别三个混着用,除非你非常了解源代码中三个混着用怎么去排序的。
ExcelIgnore导入默认所有字段都会和excel去匹配,加了这个注解会忽略该字段
DateTimeFormat导入日期转换,用String去接收excel日期格式的数据会调用这个注解。里面的value参照java.text.SimpleDateFormat
NumberFormat导入数字转换,用String去接收excel数字格式的数据会调用这个注解。里面的value参照java.text.DecimalFormat
ExcelIgnoreUnannotated导入默认不加ExcelProperty 的注解的都会参与读写,加了不会参与

导入方法参数:ReadWorkbook,ReadSheet 都会有的参数,如果为空,默认使用上级。

  • converter 转换器,默认加载了很多转换器。也可以自定义。
  • readListener 监听器,在读取数据的过程中会不断的调用监听器。
  • headRowNumber 需要读的表格有几行头数据。默认有一行头,也就是认为第二行开始起为数据。
  • headclazz二选一。读取文件头对应的列表,会根据列表匹配数据,建议使用class。
  • clazzhead二选一。读取文件的头对应的class,也可以使用注解。如果两个都不指定,则会读取全部数据。
  • autoTrim 字符串、表头等数据自动trim
  • password 读的时候是否需要使用密码

ReadWorkbook(理解成excel对象)参数

  • excelType 当前excel的类型 默认会自动判断

  • inputStreamfile二选一。读取文件的流,如果接收到的是流就只用,不用流建议使用file参数。因为使用了inputStream easyexcel会帮忙创建临时文件,最终还是file

  • fileinputStream二选一。读取文件的文件。

  • autoCloseStream 自动关闭流。

  • readCache 默认小于5M用 内存,超过5M会使用 EhCache,这里不建议使用这个参数。

  • useDefaultListener@since 2.1.4默认会加入ModelBuildEventListener来帮忙转换成传入class的对象,设置成false后将不会协助转换对象,自定义的监听器会接收到Map<Integer,CellData>对象,如果还想继续接听到

    class对象,请调用readListener方法,加入自定义的beforeListener、ModelBuildEventListener、 自定义afterListener即可。

ReadSheet(就是excel的一个Sheet)参数

  • sheetNo 需要读取Sheet的编码,建议使用这个来指定读取哪个Sheet
  • sheetName 根据名字去匹配Sheet,excel 2003不支持根据名字去匹配

添加pom依赖

<dependency><groupId>com.alibaba</groupId><artifactId>easyexcel</artifactId><version>2.2.6</version>
</dependency>
<!--工具类-->
<dependency><groupId>cn.hutool</groupId><artifactId>hutool-all</artifactId><version>5.8.23</version>
</dependency>
<dependency><groupId>com.alibaba</groupId><artifactId>fastjson</artifactId><version>1.2.21</version>
</dependency>
<!--commons依赖  -->
<dependency><groupId>org.apache.commons</groupId><artifactId>commons-lang3</artifactId><version>3.7</version>
</dependency>

第一种:简单导入

实体类

package com.example.mybatismysql8demo.excel;import com.alibaba.excel.annotation.ExcelIgnoreUnannotated;
import com.alibaba.excel.annotation.ExcelProperty;
import lombok.Data;import java.math.BigDecimal;@Data
//忽视无注解的字段
@ExcelIgnoreUnannotated
public class GoodsImportExcel {/*** 用名字去匹配,这里需要注意,如果名字重复,会导致只有一个字段读取到数据*/@ExcelProperty(value = {"商品信息","商品名称"},index = 0)public String goodsName;@ExcelProperty(value = {"商品信息","商品价格"},index = 1)public BigDecimal price;@ExcelProperty(value = {"商品信息","商品数量"},index = 2)public Integer num;}

监听器

package com.example.mybatismysql8demo.handler;import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import com.example.mybatismysql8demo.excel.GoodsImportExcel;
import com.google.common.collect.Lists;
import org.apache.commons.collections4.CollectionUtils;import java.util.List;
import java.util.function.Consumer;/*** 读取excel数据*/
public class DemoDataListener extends AnalysisEventListener<GoodsImportExcel> {/**临时存储正常数据集合,最大存储100*/private List<GoodsImportExcel> successDataList = Lists.newArrayListWithExpectedSize(100);/**自定义消费者函数接口用于自定义监听器中数据组装*/private final Consumer<List<GoodsImportExcel>> successConsumer;public DemoDataListener(Consumer<List<GoodsImportExcel>> successConsumer) {this.successConsumer = successConsumer;}@Overridepublic void invoke(GoodsImportExcel goodsImportExcel, AnalysisContext analysisContext) {successDataList.add(goodsImportExcel);System.out.println("数据:"+goodsImportExcel);}@Overridepublic void doAfterAllAnalysed(AnalysisContext analysisContext) {if (CollectionUtils.isNotEmpty(successDataList)) {successConsumer.accept(successDataList);}}
}

执行方法

package com.example.mybatismysql8demo.controller;import com.alibaba.excel.EasyExcel;
import com.alibaba.excel.ExcelReader;
import com.alibaba.excel.read.metadata.ReadSheet;
import com.alibaba.fastjson.JSONObject;
import com.example.mybatismysql8demo.excel.GoodsImportExcel;
import com.example.mybatismysql8demo.handler.DemoDataListener;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;import java.io.*;
import java.util.*;@Slf4j
@RestController
public class EasyExcelController {@PostMapping("/easyExcelImport")public void importExcel(MultipartFile file,Integer type) {if (!file.isEmpty()) {//文件名称int begin = Objects.requireNonNull(file.getOriginalFilename()).indexOf(".");//文件名称长度int last = file.getOriginalFilename().length();//判断文件格式是否正确String fileName = file.getOriginalFilename().substring(begin, last);if (!fileName.endsWith(".xls") && !fileName.endsWith(".xlsx")) {throw new IllegalArgumentException("上传文件格式错误");}} else {throw new IllegalArgumentException("文件不能为空");}try (InputStream inputStream = file.getInputStream()) {if (type == 1){simpleRead(inputStream);}else if (type == 2){synchronousRead(inputStream);}else {repeatedRead(inputStream);}} catch (IOException e) {System.out.println(e.getMessage());}}/*** 最简单的读的监听器*/public void simpleRead(InputStream inputStream){//获取正确数据ArrayList<GoodsImportExcel> successArrayList = new ArrayList<>();EasyExcel.read(inputStream).head(GoodsImportExcel.class).registerReadListener(new DemoDataListener(// 监听器中doAfterAllAnalysed执行此方法;所有读取完成之后处理逻辑successArrayList::addAll))// 设置sheet,默认读取第一个.sheet()// 设置标题(字段列表)所在行数.headRowNumber(2).doReadSync();System.out.println(successArrayList);}/*** 同步的返回,不推荐使用,如果数据量大会把数据放到内存里面*/public void synchronousRead(InputStream inputStream){// 这里 需要指定读用哪个class去读,然后读取第一个sheet 同步读取会自动finishList<GoodsImportExcel> batchGoodsImportModels = EasyExcel.read(inputStream).head(GoodsImportExcel.class)// 设置sheet,默认读取第一个.sheet()// 设置标题(字段列表)所在行数.headRowNumber(2).doReadSync();System.out.println(JSONObject.toJSONString(batchGoodsImportModels));}/*** 读取多个sheet*/public void repeatedRead(InputStream inputStream){ArrayList<GoodsImportExcel> successArrayList = new ArrayList<>();//使用模型来读取Excel(多个sheet)ExcelReader reader = EasyExcel.read(inputStream).build();//多个sheetList<ReadSheet> sheetList = new ArrayList<>();for (int i = 0; i < reader.getSheets().size(); i++){// 这里为了简单,所以注册了同样的head 和Listener 自己使用功能必须不同的ListenerReadSheet readSheet = EasyExcel.readSheet(i).head(GoodsImportExcel.class).registerReadListener(new DemoDataListener(successArrayList::addAll))// 设置标题(字段列表)所在行数.headRowNumber(2).build();sheetList.add(readSheet);}// 这里注意 一定要把sheet1 sheet2 一起传进去,不然有个问题就是03版的excel 会读取多次,浪费性能reader.read(sheetList);// 这里千万别忘记关闭,读的时候会创建临时文件,到时磁盘会崩的reader.finish();System.out.println(successArrayList);}}结果打印
数据:GoodsImportExcel(goodsName=苹果, price=10, num=11)
数据:GoodsImportExcel(goodsName=香蕉, price=8, num=12)
数据:GoodsImportExcel(goodsName=梨子, price=11.0, num=30)
数据:GoodsImportExcel(goodsName=葡萄, price=20.0, num=40)
[GoodsImportExcel(goodsName=苹果, price=10, num=11), GoodsImportExcel(goodsName=香蕉, price=8, num=12), GoodsImportExcel(goodsName=梨子, price=11.0, num=30), GoodsImportExcel(goodsName=葡萄, price=20.0, num=40)]

导入模版
在这里插入图片描述
在这里插入图片描述

第二种:数据校验

自定义注解

package com.example.mybatismysql8demo.config;import java.lang.annotation.*;@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
@Inherited
public @interface LengthValid {int length() default 0;String msg() default "";int cell() default 0;
}

实体类

package com.example.mybatismysql8demo.excel;import com.alibaba.excel.annotation.ExcelIgnoreUnannotated;
import com.alibaba.excel.annotation.ExcelProperty;
import com.example.mybatismysql8demo.config.LengthValid;
import lombok.Data;import java.math.BigDecimal;@Data
//忽视无注解的字段
@ExcelIgnoreUnannotated
public class GoodsImportExcel {/*** 用名字去匹配,这里需要注意,如果名字重复,会导致只有一个字段读取到数据*/@LengthValid(length =  5,msg = "商品名称长度超出5个字符串!",cell = 1)@ExcelProperty(value = {"商品信息","商品名称"},index = 0)public String goodsName;@ExcelProperty(value = {"商品信息","商品价格"},index = 1)public BigDecimal price;@ExcelProperty(value = {"商品信息","商品数量"},index = 2)public Integer num;private String errorMsg;
}

监听器

package com.example.mybatismysql8demo.handler;import com.alibaba.excel.annotation.ExcelProperty;
import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import com.alibaba.excel.exception.ExcelDataConvertException;
import com.alibaba.excel.metadata.CellData;
import com.alibaba.excel.metadata.CellExtra;
import com.alibaba.fastjson.JSONObject;
import com.baomidou.mybatisplus.extension.api.Assert;
import com.example.mybatismysql8demo.config.LengthValid;
import com.example.mybatismysql8demo.excel.GoodsImportExcel;
import com.google.common.collect.Lists;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.collections4.CollectionUtils;import java.lang.reflect.Field;
import java.util.List;
import java.util.Map;
import java.util.function.Consumer;/*** 读取excel数据*/
@Slf4j
public class DemoDataListener extends AnalysisEventListener<GoodsImportExcel> {/**单次处理上限100条记录*/private static final int BATCH_COUNT = 100;/**临时存储正常数据集合*/private List<GoodsImportExcel> successDataList = Lists.newArrayListWithExpectedSize(BATCH_COUNT);/**临时存错误储数据集合*/private List<GoodsImportExcel> errorDataList = Lists.newArrayListWithExpectedSize(BATCH_COUNT);/**自定义消费者函数接口用于自定义监听器中数据组装*/private final Consumer<List<GoodsImportExcel>> successConsumer;private final Consumer<List<GoodsImportExcel>> errorConsumer;public DemoDataListener(Consumer<List<GoodsImportExcel>> successConsumer, Consumer<List<GoodsImportExcel>> errorConsumer) {this.successConsumer = successConsumer;this.errorConsumer = errorConsumer;}/**手机号格式异常日志处理*/@Overridepublic void onException(Exception exception, AnalysisContext context) {log.error("异常信息:{}", exception.getMessage());// 如果是某一个单元格的转换异常 能获取到具体行号,如果要获取头的信息 配合invokeHeadMap使用if (exception instanceof ExcelDataConvertException) {ExcelDataConvertException excelDataConvertException = (ExcelDataConvertException) exception;log.error("第{}行,第{}列解析异常,数据为:{}", excelDataConvertException.getRowIndex(), excelDataConvertException.getColumnIndex(), excelDataConvertException.getCellData());}else if (exception instanceof IllegalArgumentException){throw new IllegalArgumentException(exception.getMessage());}}/*** 在这里进行模板的判断* @param headMap 存放着导入表格的表头,键是索引,值是名称* @param context*/@Overridepublic void invokeHeadMap(Map<Integer, String> headMap, AnalysisContext context) {//只校验第三行表头是否正确Integer rowNum = context.getCurrentRowNum();if (rowNum == 2) {// 获取数据实体的字段列表Field[] fields = GoodsImportExcel.class.getDeclaredFields();// 遍历字段进行判断for (Field field : fields) {// 获取当前字段上的ExcelProperty注解信息ExcelProperty fieldAnnotation = field.getAnnotation(ExcelProperty.class);// 判断当前字段上是否存在ExcelProperty注解if (fieldAnnotation != null) {String value = fieldAnnotation.value()[1];// 存在ExcelProperty注解则根据注解的value值到表格中对比是否存在对应的表头if(!headMap.containsValue(value)){// 如果表格不包含模版类字段中的表头,则抛出异常不再往下执行throw new RuntimeException("模板错误,请检查导入模板");}}}}}/**每行读取监听触发逻辑*/@SneakyThrows@Overridepublic void invoke(GoodsImportExcel goodsImportExcel, AnalysisContext analysisContext) {//获取总行数Integer rowNumber = analysisContext.readSheetHolder().getApproximateTotalRowNumber();//行数int row = analysisContext.readRowHolder().getRowIndex();log.info("第" + row + "行数据进行处理");// 手机号格式校验validParam(goodsImportExcel,row);//正常数据successDataList.add(goodsImportExcel);// 按照指定条数对导入数据进行分批处理if (successDataList.size() >= BATCH_COUNT) {successConsumer.accept(successDataList);successDataList = Lists.newArrayListWithExpectedSize(BATCH_COUNT);}}private void validParam(GoodsImportExcel goodsImportExcel, int row) throws IllegalAccessException {// 参数校验Field[] fields = goodsImportExcel.getClass().getDeclaredFields();for (Field field : fields) {//设置可访问field.setAccessible(true);//判断字段是否添加校验boolean valid = field.isAnnotationPresent(LengthValid.class);if (valid) {//获取注解信息LengthValid annotation = field.getAnnotation(LengthValid.class);//行数列数String msg = "第" + row + "行的第" + annotation.cell() + "列:";//值String value = (String) field.get(goodsImportExcel);if(value.length() > annotation.length()){//错误信息goodsImportExcel.setErrorMsg(msg + annotation.msg());//错误数据errorDataList.add(goodsImportExcel);// 按照指定条数对导入数据进行分批处理if (errorDataList.size() >= BATCH_COUNT) {errorConsumer.accept(errorDataList);errorDataList = Lists.newArrayListWithExpectedSize(BATCH_COUNT);}throw new RuntimeException(msg + annotation.msg());}}}}@Overridepublic void doAfterAllAnalysed(AnalysisContext analysisContext) {if (CollectionUtils.isNotEmpty(successDataList)) {successConsumer.accept(successDataList);}if (CollectionUtils.isNotEmpty(errorDataList)) {errorConsumer.accept(errorDataList);}}/*** 额外信息(批注、超链接、合并单元格信息读取)*/@Overridepublic void extra(CellExtra extra, AnalysisContext context) {log.info("读取到了一条额外信息:{}", JSONObject.toJSONString(extra));switch (extra.getType()) {case COMMENT:log.info("额外信息是批注,在rowIndex:{},columnIndex;{},内容是:{}", extra.getRowIndex(), extra.getColumnIndex(), extra.getText());break;case HYPERLINK:if ("Sheet1!A1".equals(extra.getText())) {log.info("额外信息是超链接,在rowIndex:{},columnIndex;{},内容是:{}", extra.getRowIndex(), extra.getColumnIndex(), extra.getText());} else if ("Sheet2!A1".equals(extra.getText())) {log.info("额外信息是超链接,而且覆盖了一个区间,在firstRowIndex:{},firstColumnIndex;{},lastRowIndex:{},lastColumnIndex:{}," + "内容是:{}",extra.getFirstRowIndex(), extra.getFirstColumnIndex(), extra.getLastRowIndex(),extra.getLastColumnIndex(), extra.getText());} else {Assert.fail("Unknown hyperlink!");}break;case MERGE:log.info("额外信息是超链接,而且覆盖了一个区间,在firstRowIndex:{},firstColumnIndex;{},lastRowIndex:{},lastColumnIndex:{}",extra.getFirstRowIndex(), extra.getFirstColumnIndex(), extra.getLastRowIndex(),extra.getLastColumnIndex());break;default:}}/***监听器的hasNext()方法时没有注意到默认返回的是false,导致一进监听器就判断已经没有下一条记录,直接跳出监听器,然后导入就完成,也不会报错,改成返回true即可解决*/@Overridepublic boolean hasNext(AnalysisContext analysisContext) {return true;}
}

执行方法

package com.example.mybatismysql8demo.controller;import com.alibaba.excel.EasyExcel;
import com.alibaba.excel.ExcelReader;
import com.alibaba.excel.read.metadata.ReadSheet;
import com.alibaba.fastjson.JSONObject;
import com.example.mybatismysql8demo.excel.GoodsImportExcel;
import com.example.mybatismysql8demo.handler.DemoDataListener;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;import java.io.*;
import java.util.*;@Slf4j
@RestController
public class EasyExcelController {@PostMapping("/easyExcelImport")public void importExcel(MultipartFile file,Integer type) {if (!file.isEmpty()) {//文件名称int begin = Objects.requireNonNull(file.getOriginalFilename()).indexOf(".");//文件名称长度int last = file.getOriginalFilename().length();//判断文件格式是否正确String fileName = file.getOriginalFilename().substring(begin, last);if (!fileName.endsWith(".xls") && !fileName.endsWith(".xlsx")) {throw new IllegalArgumentException("上传文件格式错误");}} else {throw new IllegalArgumentException("文件不能为空");}try (InputStream inputStream = file.getInputStream()) {if (type == 1){simpleRead(inputStream);}else if (type == 2){synchronousRead(inputStream);}else {repeatedRead(inputStream);}} catch (IOException e) {System.out.println(e.getMessage());}}/*** 最简单的读的监听器*/public void simpleRead(InputStream inputStream){//获取正确数据ArrayList<GoodsImportExcel> successArrayList = new ArrayList<>();//获取错误数据ArrayList<GoodsImportExcel> errorArrayList = new ArrayList<>();EasyExcel.read(inputStream).head(GoodsImportExcel.class).registerReadListener(new DemoDataListener(// 监听器中doAfterAllAnalysed执行此方法;所有读取完成之后处理逻辑successArrayList::addAll, errorArrayList::addAll))// 设置sheet,默认读取第一个.sheet()// 设置标题(字段列表)所在行数.headRowNumber(2).doReadSync();System.out.println(successArrayList);System.out.println(errorArrayList);}/*** 同步的返回,不推荐使用,如果数据量大会把数据放到内存里面*/public void synchronousRead(InputStream inputStream){// 这里 需要指定读用哪个class去读,然后读取第一个sheet 同步读取会自动finishList<GoodsImportExcel> batchGoodsImportModels = EasyExcel.read(inputStream).head(GoodsImportExcel.class)// 设置sheet,默认读取第一个.sheet()// 设置标题(字段列表)所在行数.headRowNumber(2).doReadSync();System.out.println(JSONObject.toJSONString(batchGoodsImportModels));}/*** 读取多个sheet*/public void repeatedRead(InputStream inputStream){ArrayList<GoodsImportExcel> successArrayList = new ArrayList<>();//获取错误数据ArrayList<GoodsImportExcel> errorArrayList = new ArrayList<>();//使用模型来读取Excel(多个sheet)ExcelReader reader = EasyExcel.read(inputStream).build();//多个sheetList<ReadSheet> sheetList = new ArrayList<>();for (int i = 0; i < reader.getSheets().size(); i++){// 这里为了简单,所以注册了同样的head 和Listener 自己使用功能必须不同的ListenerReadSheet readSheet = EasyExcel.readSheet(i).head(GoodsImportExcel.class).registerReadListener(new DemoDataListener(successArrayList::addAll, errorArrayList::addAll))// 设置标题(字段列表)所在行数.headRowNumber(2).build();sheetList.add(readSheet);}// 这里注意 一定要把sheet1 sheet2 一起传进去,不然有个问题就是03版的excel 会读取多次,浪费性能reader.read(sheetList);// 这里千万别忘记关闭,读的时候会创建临时文件,到时磁盘会崩的reader.finish();System.out.println(successArrayList);System.out.println(errorArrayList);}
}结果打印
[GoodsImportExcel(goodsName=苹果, price=10, num=11, errorMsg=null), GoodsImportExcel(goodsName=香蕉, price=8, num=12, errorMsg=null), GoodsImportExcel(goodsName=葡萄, price=20.0, num=40, errorMsg=null)]
[GoodsImportExcel(goodsName=梨子1111, price=11.0, num=30, errorMsg=2行的第1:商品名称长度超出5个字符串!)]

导入模版
在这里插入图片描述

第三种:读取存在合并

监听器

package com.example.mybatismysql8demo.handler;
import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import com.alibaba.excel.metadata.CellExtra;
import com.alibaba.fastjson.JSON;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.util.ArrayList;
import java.util.List;/*** Excel模板的读取监听类* @author gd*/
public class ImportExcelListener<T> extends AnalysisEventListener<T> {private static final Logger LOGGER = LoggerFactory.getLogger(ImportExcelListener.class);/*** 解析的数据*/private final List<T> list = new ArrayList<>();/*** 正文起始行*/private final Integer headRowNumber;/*** 合并单元格*/private final List<CellExtra> extraMergeInfoList = new ArrayList<>();public ImportExcelListener(Integer headRowNumber) {this.headRowNumber = headRowNumber;}/*** 这个每一条数据解析都会来调用*/@Overridepublic void invoke(T data, AnalysisContext context) {LOGGER.info("数据处理: " + JSON.toJSONString(data));list.add(data);}/*** 所有数据解析完成了 都会来调用* @param context context*/@Overridepublic void doAfterAllAnalysed(AnalysisContext context) {LOGGER.info("所有数据解析完成!");}/*** 返回解析出来的List*/public List<T> getData() {return list;}/*** 读取额外信息:合并单元格*/@Overridepublic void extra(CellExtra extra, AnalysisContext context) {LOGGER.info("读取到了一条额外信息:{}", JSON.toJSONString(extra));switch (extra.getType()) {case COMMENT:LOGGER.info("额外信息是批注,在rowIndex:{},columnIndex;{},内容是:{}", extra.getRowIndex(), extra.getColumnIndex(), extra.getText());break;case MERGE: {LOGGER.info("额外信息是合并单元格,而且覆盖了一个区间,在firstRowIndex:{},firstColumnIndex;{},lastRowIndex:{},lastColumnIndex:{}",extra.getFirstRowIndex(), extra.getFirstColumnIndex(), extra.getLastRowIndex(), extra.getLastColumnIndex());if (extra.getRowIndex() >= headRowNumber) {extraMergeInfoList.add(extra);}break;}default:}}/*** 返回解析出来的合并单元格List*/public List<CellExtra> getExtraMergeInfoList() {return extraMergeInfoList;}
}

合并数据处理工具类

package com.example.mybatismysql8demo.utils;import com.alibaba.excel.EasyExcel;
import com.alibaba.excel.annotation.ExcelProperty;
import com.alibaba.excel.enums.CellExtraTypeEnum;
import com.alibaba.excel.metadata.CellExtra;
import com.alibaba.excel.util.CollectionUtils;
import com.example.mybatismysql8demo.config.LengthValid;
import com.example.mybatismysql8demo.excel.GoodsImportExcel;
import com.example.mybatismysql8demo.handler.ImportExcelListener;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;import java.io.InputStream;
import java.lang.reflect.Field;
import java.util.List;public class ImportExcelMergeUtil<T> {private static final Logger LOGGER = LoggerFactory.getLogger(ImportExcelMergeUtil.class);/*** 返回解析后的List** @param: fileName 文件名* @param: clazz Excel对应属性名* @param: sheetNo 要解析的sheet* @param: headRowNumber 正文起始行* @return java.util.List<T> 解析后的List*/public void getList(InputStream inputStream, Class<GoodsImportExcel> clazz, Integer sheetNo, Integer headRowNumber,List<T> successList,List<T> errorList) {ImportExcelListener<T> listener = new ImportExcelListener<>(headRowNumber);try {EasyExcel.read(inputStream, clazz, listener).extraRead(CellExtraTypeEnum.MERGE).sheet(sheetNo).headRowNumber(headRowNumber).doRead();} catch (Exception e) {LOGGER.error(e.getMessage());}List<CellExtra> extraMergeInfoList = listener.getExtraMergeInfoList();//解析数据List<T> list;if (CollectionUtils.isEmpty(extraMergeInfoList)) {list = (listener.getData());}else {list = explainMergeData(listener.getData(), extraMergeInfoList, headRowNumber);}//数据处理for (T v : list) {if(validParam(v)){errorList.add(v);}else {successList.add(v);}}}private Boolean validParam(T object){// 参数校验Field[] fields = object.getClass().getDeclaredFields();for (Field field : fields) {//设置可访问field.setAccessible(true);//判断字段是否添加校验boolean valid = field.isAnnotationPresent(LengthValid.class);if (valid) {try {//获取注解信息LengthValid annotation = field.getAnnotation(LengthValid.class);//值String value = (String) field.get(object);if(value.length() > annotation.length()){//错误信息(需要设置字段为public)Field errorMsg = object.getClass().getField("errorMsg");if (errorMsg.get(object) == null){errorMsg.set(object, annotation.msg());}else {errorMsg.set(object,errorMsg.get(object) + "," + annotation.msg());}return true;}} catch (IllegalAccessException | NoSuchFieldException e) {e.printStackTrace();}}}return false;}/*** 处理合并单元格* @param data               解析数据* @param extraMergeInfoList 合并单元格信息* @param headRowNumber      起始行* @return 填充好的解析数据*/private List<T> explainMergeData(List<T> data, List<CellExtra> extraMergeInfoList, Integer headRowNumber) {//循环所有合并单元格信息extraMergeInfoList.forEach(cellExtra -> {int firstRowIndex = cellExtra.getFirstRowIndex() - headRowNumber;int lastRowIndex = cellExtra.getLastRowIndex() - headRowNumber;int firstColumnIndex = cellExtra.getFirstColumnIndex();int lastColumnIndex = cellExtra.getLastColumnIndex();//获取初始值Object initValue = getInitValueFromList(firstRowIndex, firstColumnIndex, data);//设置值for (int i = firstRowIndex; i <= lastRowIndex; i++) {for (int j = firstColumnIndex; j <= lastColumnIndex; j++) {setInitValueToList(initValue, i, j, data);}}});return data;}/*** 设置合并单元格的值** @param filedValue  值* @param rowIndex    行* @param columnIndex 列* @param data        解析数据*/private void setInitValueToList(Object filedValue, Integer rowIndex, Integer columnIndex, List<T> data) {T object = data.get(rowIndex);for (Field field : object.getClass().getDeclaredFields()) {//提升反射性能,关闭安全检查field.setAccessible(true);ExcelProperty annotation = field.getAnnotation(ExcelProperty.class);if (annotation != null) {if (annotation.index() == columnIndex) {try {field.set(object, filedValue);break;} catch (IllegalAccessException e) {LOGGER.error("设置合并单元格的值异常:"+e.getMessage());}}}}}/*** 获取合并单元格的初始值* rowIndex对应list的索引* columnIndex对应实体内的字段* @param firstRowIndex    起始行* @param firstColumnIndex 起始列* @param data             列数据* @return 初始值*/private Object getInitValueFromList(Integer firstRowIndex, Integer firstColumnIndex, List<T> data) {Object filedValue = null;T object = data.get(firstRowIndex);for (Field field : object.getClass().getDeclaredFields()) {//提升反射性能,关闭安全检查field.setAccessible(true);ExcelProperty annotation = field.getAnnotation(ExcelProperty.class);if (annotation != null) {if (annotation.index() == firstColumnIndex) {try {filedValue = field.get(object);break;} catch (IllegalAccessException e) {LOGGER.error("获取合并单元格的初始值异常:"+e.getMessage());}}}}return filedValue;}
}

执行方法

package com.example.mybatismysql8demo.controller;import com.alibaba.fastjson.JSONObject;
import com.example.mybatismysql8demo.excel.GoodsImportExcel;
import com.example.mybatismysql8demo.utils.ImportExcelMergeUtil;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.multipart.MultipartFile;import java.io.*;
import java.util.*;@Slf4j
@RestController
public class EasyExcelController {@PostMapping("/easyExcelImport")public void importExcel(MultipartFile file) {if (!file.isEmpty()) {//文件名称int begin = Objects.requireNonNull(file.getOriginalFilename()).indexOf(".");//文件名称长度int last = file.getOriginalFilename().length();//判断文件格式是否正确String fileName = file.getOriginalFilename().substring(begin, last);if (!fileName.endsWith(".xls") && !fileName.endsWith(".xlsx")) {throw new IllegalArgumentException("上传文件格式错误");}} else {throw new IllegalArgumentException("文件不能为空");}ImportExcelMergeUtil<GoodsImportExcel> helper = new ImportExcelMergeUtil<>();List<GoodsImportExcel> successList = new ArrayList<>();List<GoodsImportExcel> errorList = new ArrayList<>();try {helper.getList(file.getInputStream(), GoodsImportExcel.class,0,2,successList,errorList);System.out.println("正确数据:"+successList);System.out.println("错误数据:"+errorList);} catch (IOException e) {e.printStackTrace();}}
}结果打印
正确数据:[GoodsImportExcel(goodsName=香蕉, price=8, num=12, errorMsg=null)]
错误数据:[GoodsImportExcel(goodsName=苹果11111, price=10, num=11, errorMsg=商品名称长度超出5个字符串!), GoodsImportExcel(goodsName=苹果11111, price=9.0, num=20, errorMsg=商品名称长度超出5个字符串!)]

导入模版
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/829193.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

个人电脑本地部署LLM

普通电脑配置即可本地运行大模型&#xff0c;本地部署LLM最简单的方法 OLLAMA Ollama是一个开源框架&#xff0c;专门设计用于在本地运行大型语言模型&#xff08;LLM&#xff09;。它的主要功能是在Docker容器中部署和管理LLM&#xff0c;使得在本地运行大模型的过程变得非常…

python之List列表

1. 高级数据类型 Python中的数据类型可以分为&#xff1a;数字型&#xff08;基本数据类型&#xff09;和非数字型&#xff08;高级数据类型&#xff09; 数字型包含&#xff1a;整型int、浮点型float、布尔型bool、复数型complex 非数字型包含&#xff1a;字符串str、列表l…

CPPTest实例分析(C++ Test)

1 概述 CppTest是一个可移植、功能强大但简单的单元测试框架&#xff0c;用于处理C中的自动化测试。重点在于可用性和可扩展性。支持多种输出格式&#xff0c;并且可以轻松添加新的输出格式。 CppTest下载地址&#xff1a;下载地址1  下载地址2 下面结合实例分析下CppTest如…

Vue3+ts(day03:ref和reactive)

学习源码可以看我的个人前端学习笔记 (github.com):qdxzw/frontlearningNotes 觉得有帮助的同学&#xff0c;可以点心心支持一下哈&#xff08;笔记是根据b站上学习的尚硅谷的前端视频【张天禹老师】&#xff0c;记录一下学习笔记&#xff0c;用于自己复盘&#xff0c;有需要学…

ARCGIS PRO3 三维模型OSGB转SLPK场景数据集

1.前言 因项目工作&#xff0c;需要将三维模型发布到arcgisserver上&#xff0c;但arcgisserver只支持slpk格式的模型&#xff0c;于是我开启了漫长的三维模型格式转换之旅&#xff0c;在这里记录下本人踩过的坑。 2.三维模型数据情况 2.1 模型大小&#xff1a;在20GB以上&a…

tcp inflight 守恒算法的自动收敛

inflight 守恒算法看起来只描述理想情况&#xff0c;现实很难满足&#xff0c;是这样吗&#xff1f; 从 reno 到 bbr&#xff0c;无论哪个算法都在描述理想情况&#xff0c;以 reno 和 bbr 两个极端为例&#xff0c;它们分别描述两种理想管道&#xff0c;reno 将 buffer 从恰好…

goroutinue和channel

goroutinue和channel 需求传统方式实现goroutinue进程和线程说明并发和并行go协程和go主线程MPG设置Go运行的cpu数 channel(管道)-看个需求使用互斥锁、写锁channel 实现 使用select可以解决从管道取数据的阻塞问题&#xff08;无需手动关闭channel了&#xff09;goroutinue中使…

61、回溯-分割回文串

思路&#xff1a; 还是全排列的思路&#xff0c;列出每一种组合&#xff0c;然后验证是否是回文&#xff0c;如果是子串放入path中&#xff0c;在验证其他元素是否也是回文。代码如下&#xff1a; class Solution {// 主方法&#xff0c;用于接收一个字符串s并返回所有可能的…

智能变频三模正弦波控制器

智能变频三模正弦波控制器 前言一、图片介绍总结 前言 不敢动&#xff0c;完全不敢动。多做笔记&#xff0c;完全了解之后再说吧 一、图片介绍 轮毂电机 主角登场 淘宝关于这款控制器的介绍 当然不同的型号功能不同 学习线插上就会转,可以使用继电器控制通断。 电门…

mac资源库的东西可以删除吗?提升Mac运行速度秘籍 Mac实用软件

很多小伙伴在使用mac电脑处理工作的时候&#xff0c;就会很疑惑&#xff0c;电脑的运行速度怎么越来越慢&#xff0c;就想着通过删除mac资源库的东西&#xff0c;那么mac资源库的东西可以删除吗&#xff1f;删除了会不会造成电脑故障呢&#xff1f; 首先&#xff0c;mac资源库…

解决ax = Axes3D(fig2)pycharm画3d图空白不显示问题

明明代码运行正确&#xff0c;却总是显示不出来 绘制出来的也是空白 改一下代码就好了 ax Axes3D(fig2) #原来代码 ax fig2.add_axes(Axes3D(fig2)) #改后代码 修改过后就可以显示了

深入了解MySQL:从基础到特性,全面解读关系数据库管理系统的历史与应用

文章目录 1. MySQL简介1.1 概述1.2 架构与兼容性1.3 开源与社区支持 2. MySQL的历史2.1 创始与初衷2.2 发展历程2.3 在Oracle的持续发展2.4 开源与商业结合 3. MySQL的核心特性4. MySQL在实际应用中的作用4.1 网站建设与内容管理4.2 商业智能与客户关系管理4.3 企业级应用与云集…

线性代数 --- 计算斐波那契数列第n项的快速算法(矩阵的n次幂)

计算斐波那契数列第n项的快速算法(矩阵的n次幂) The n-th term of Fibonacci Numbers&#xff1a; 斐波那契数列的是一个古老而又经典的数学数列&#xff0c;距今已经有800多年了。关于斐波那契数列的计算方法不难&#xff0c;只是当我们希望快速求出其数列中的第100&#xff0…

mysql8.0免安装版windows

1.下载 MySQL下载链接 2.解压与新建my.ini文件 解压的路径最好不要有中文路径在\mysql-8.0.36-winx64文件夹下新建my.ini文件&#xff0c;不建data文件夹(会自动生成) [mysqld] # 设置3306端口 port3306 # 设置mysql的安装目录(尽量用双斜杠\\,单斜杠\可能会报错) basedirD:\…

uniapp获取当前位置及检测授权状态

uniapp获取当前位置及检测授权定位权限 文章目录 uniapp获取当前位置及检测授权定位权限效果图创建js文件permission.jslocation.js 使用 效果图 Android设备 点击 “设置”&#xff0c;跳转应用信息&#xff0c;打开“权限即可”&#xff1b; 创建js文件 permission.js 新建…

3d合并的模型为什么没有模型---模大狮模型网

在3D建模中&#xff0c;合并模型是常见的操作&#xff0c;它可以将多个模型合并成一个整体。然而&#xff0c;有时候在合并后却发现部分模型消失了&#xff0c;这可能会让人感到困惑和失望。本文将探讨为什么合并的3D模型中会出现没有模型的情况&#xff0c;并提供一些解决方法…

Web前端一套全部清晰 ③ day2 HTML 标签综合案例

别让平淡生活&#xff0c;耗尽所有向往 —— 24.4.26 综合案例 —— 一切都会好的 网页制作思路&#xff1a;从上到下&#xff0c;先整体到局部&#xff0c;逐步分析制作 分析内容 ——> 写代码 ——>保存——>刷新浏览器&#xff0c;看效果 <!DOCTYPE html> &l…

ubuntu22 部署fastDFS单节点和集群,整合Spring Boot(刚部署成功)

ubuntu22 部署fastDFS单节点和集群 一、先准备1、所需依赖安装2、下载安装包 二、安装FastDFS单节点1、libfastcommon安装1.1、创建软连接 2、安装fastDFS2.1、fastDFS目录简单介绍2.2、创建软连接 3、配置和启动Tracker服务3.1、修改Tracker配置文件3.2、启动Tracker 4、配置和…

【笔试强训】除2!

登录—专业IT笔试面试备考平台_牛客网牛客网是互联网求职神器&#xff0c;C、Java、前端、产品、运营技能学习/备考/求职题库&#xff0c;在线进行百度阿里腾讯网易等互联网名企笔试面试模拟考试练习,和牛人一起讨论经典试题,全面提升你的技术能力https://ac.nowcoder.com/acm/…

IDEA插件分享 - enum-quick-generate 实现枚举类自动生成

&#x1f604; 19年之后由于某些原因断更了三年&#xff0c;23年重新扬帆起航&#xff0c;推出更多优质博文&#xff0c;希望大家多多支持&#xff5e; &#x1f337; 古之立大事者&#xff0c;不惟有超世之才&#xff0c;亦必有坚忍不拔之志 &#x1f390; 个人CSND主页——Mi…