EasyExcel实现复杂Excel的导入

最近项目中遇到一个复杂的Excel的导入,并且数据量较大。因为数据不规则,所以只能使用POI进行自定义读取,但是发现数据量大之后,读取数据非常耗时。后面换成EasyExcel,性能起飞。

1. Excel样板

excel样例
如上图,需要导入学校学生信息。前面三列,固定为学校班级信息,可以理解为主表。从第四列开始,为学生信息,其中一列为一条子表信息。

2. 具体实现

  1. 引入EasyExcel的依赖
<dependency>
	<groupId>com.alibaba</groupId>
	<artifactId>easyexcel</artifactId>
	<version>3.3.2</version>
</dependency>
  1. 自定义导入监听器
import com.alibaba.excel.context.AnalysisContext;
import com.alibaba.excel.event.AnalysisEventListener;
import com.alibaba.fastjson.JSON;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;

public class ImportExcelListener extends AnalysisEventListener<Object> {
	// 封装数据	
	private List<Object> list = new ArrayList<>();
	// sheet页索引
	private int sheetNo = 0;
	
	@Override
	public void invoke(Object t,AnalysisContext context) {
		// 读取Excel内容
		int currentSheetNo = context.readSheetHolder().getSheetNo();
		if(currentSheetNo != sheetNo){
			// 根据sheet页索引,防止重复添加数据
			list = new ArrayList<>();
			sheetNo = currentSheetNo;
		}
		list.add(t);
	}
	
	/**
	 * 将表格转化为map集合(复杂Excel)
	 */
	public List<LinkedHashMap> getListMap(){
		String jsonObj = JSON.toJSONString(list);
		return JSON.parseArray(jsonObj,LinkedHashMap.class);
	}
	
	/**
	 * 读取完数据后的操作
	 */
	@Override
	public void doAfterAllAnalysed(AnalysisContext context){
	}
}
  1. 测试使用
    3.1. TestController
@RestController
public class ExcelContorller{

    @Autowired
	private ExcelService excelService;

	@PostMapping("/testImport")
	public Boolean importExcel(MultipartFile file) throws IOException {
		return excelService.importExcel(file);
	}
}

3.2. ExcelService

import com.alibaba.excel.EasyExcelFactory;
import com.alibaba.excel.ExcelReader;
import com.alibaba.excel.read.metadata.ReadSheet;
import com.test.excel.bean.SchoolInfo;
import com.test.excel.listener.ImportExcelListener;
import lombok.extern.slf4j.Slf4j;
import org.springframework.stereotype.Service;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.util.StopWatch;

import java.io.IOException;
import java.util.ArrayList;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;

@Slf4j
@Service
public class ExcelService {
	
	/**
	 * 导入
	 */
	public boolean importExcel(MultipartFile file) throws IOException {
		log.debug("进入导出方法");
		StopWatch watch = new StopWatch();
		watch.start();
		ImportExcelListener importListener = new ImportExcelListener ();
		ExcelReader excelReader = null;
		try {
			// 1为跳过标题行数
			excelReader = EasyExcelFactory.read(file.getInputStream,importListener).headRowNumber(1).build();
			List<ReadSheet> sheets = excelReader.excelExecutor().sheetList();
			List<SchoolInfo> dataList = new ArrayList<>();
			// 循环处理Sheet
			for (ReadSheet sheet : sheets){
				excelReader.read(sheet);
				this.dealSheetData(importListener.getListMap(),dataList);
			}
		}catch(Exception ex){
			log.error("导入失败",ex)
		}finally{
			watch.stop();
			log.debug("导入耗时{}毫秒",watch.getTotalTimeMilis());
		}
	}
}

/**
 * 数据处理
 */
private void dealSheetData(List<LinkedHashMap> rowList,List<SchoolInfo> resultList) {
	int rowSize = rowList.size();
	for(int i=0; i < rowSize; i++){
		// 这里为一行的数据,map的key为单元格的下标,从0开始,value为对应单元格的值
		LinkedHashMap<Integer,String> row = rowList.get(i);
		// 数据处理
		...
	}
}
  • 7
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
对于复杂Excel文件,你可以使用EasyExcel导入数据。EasyExcel是一个基于Java的开源库,可以用来读取、写入和处理Excel文件。 下面是一个示例代码,演示如何使用EasyExcel导入复杂Excel文件: ```java // 导入相关的包 import com.alibaba.excel.EasyExcel; import com.alibaba.excel.context.AnalysisContext; import com.alibaba.excel.event.AnalysisEventListener; // 创建一个数据模型类,用于存储导入的数据 public class ExcelData { private String column1; private String column2; // ... 其他列 // 省略getter和setter方法 } // 创建一个监听器类,用于处理导入的数据 public class ExcelListener extends AnalysisEventListener<ExcelData> { @Override public void invoke(ExcelData data, AnalysisContext context) { // 在这里处理每一行的数据 System.out.println("读取到数据:" + data.getColumn1() + ", " + data.getColumn2()); // ... 处理其他列 } @Override public void doAfterAllAnalysed(AnalysisContext context) { // 当所有数据都解析完成后,执行此方法 } } // 主程序 public class Main { public static void main(String[] args) { String filePath = "path/to/your/excel/file.xlsx"; // 使用EasyExcel进行导入 EasyExcel.read(filePath, ExcelData.class, new ExcelListener()).sheet().doRead(); } } ``` 在上面的示例中,你需要自己定义一个数据模型类(ExcelData),用来存储导入的数据。然后创建一个监听器类(ExcelListener),通过继承AnalysisEventListener来处理每一行的数据。最后,在主程序中使用EasyExcel的read方法来读取Excel文件并进行导入,你只需要将文件路径替换成你实际的Excel文件路径即可。 希望能帮到你!如果有更多问题,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值