|  |  | package com.glxp.api.util.Excel;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | import com.alibaba.excel.EasyExcel;
 | 
						
						
						
							|  |  | import com.alibaba.excel.EasyExcelFactory;
 | 
						
						
						
							|  |  | import com.alibaba.excel.ExcelWriter;
 | 
						
						
						
							|  |  | import com.alibaba.excel.read.builder.ExcelReaderBuilder;
 | 
						
						
						
							|  |  | import com.alibaba.excel.write.metadata.WriteSheet;
 | 
						
						
						
							|  |  | import com.alibaba.excel.write.metadata.fill.FillConfig;
 | 
						
						
						
							|  |  | import com.glxp.api.util.Excel.Exception.RenException;
 | 
						
						
						
							|  |  | import lombok.extern.slf4j.Slf4j;
 | 
						
						
						
							|  |  | import org.springframework.stereotype.Component;
 | 
						
						
						
							|  |  | import org.springframework.util.StringUtils;
 | 
						
						
						
							|  |  | import org.springframework.web.multipart.MultipartFile;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | import javax.servlet.ServletOutputStream;
 | 
						
						
						
							|  |  | import javax.servlet.http.HttpServletResponse;
 | 
						
						
						
							|  |  | import java.io.File;
 | 
						
						
						
							|  |  | import java.io.InputStream;
 | 
						
						
						
							|  |  | import java.io.OutputStream;
 | 
						
						
						
							|  |  | import java.net.URLEncoder;
 | 
						
						
						
							|  |  | import java.nio.charset.StandardCharsets;
 | 
						
						
						
							|  |  | import java.util.LinkedList;
 | 
						
						
						
							|  |  | import java.util.List;
 | 
						
						
						
							|  |  | import java.util.Map;
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | @Slf4j
 | 
						
						
						
							|  |  | @Component
 | 
						
						
						
							|  |  | public class ExcelHandler {
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 导入简单excel数据
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param file                :文件流
 | 
						
						
						
							|  |  |      * @param clazz:数据对象
 | 
						
						
						
							|  |  |      * @param sheetName:要读取的sheet [不传:默认读取第一个sheet]
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> List<T> importExcel(MultipartFile file, Class<T> clazz, String sheetName) throws Exception {
 | 
						
						
						
							|  |  |         this.checkFile(file);
 | 
						
						
						
							|  |  |         UploadDataListener<T> uploadDataListener = new UploadDataListener<>();
 | 
						
						
						
							|  |  |         ExcelReaderBuilder builder = EasyExcelFactory.read(file.getInputStream(), clazz, uploadDataListener);
 | 
						
						
						
							|  |  |         if (StringUtils.isEmpty(sheetName)) {
 | 
						
						
						
							|  |  |             builder.sheet().doRead();
 | 
						
						
						
							|  |  |         } else {
 | 
						
						
						
							|  |  |             builder.sheet(sheetName).doRead();
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |         return uploadDataListener.getList();
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 指定sheet页导入通用方法
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param multipartFile 传入文件
 | 
						
						
						
							|  |  |      * @param objList       需要导入的sheet页实体类型集合
 | 
						
						
						
							|  |  |      * @param index         sheet页个数
 | 
						
						
						
							|  |  |      * @param indexList     需要导入sheet页下标集合
 | 
						
						
						
							|  |  |      * @param <T>
 | 
						
						
						
							|  |  |      * @return <T> List<List<T>>
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> List<List<T>> importExcelsByIndex(MultipartFile multipartFile, List<T> objList, int index, List<Integer> indexList) throws Exception {
 | 
						
						
						
							|  |  |         if (multipartFile == null) {
 | 
						
						
						
							|  |  |             throw new RenException("文件为空");
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |         List<List<T>> resultList = new LinkedList<>();
 | 
						
						
						
							|  |  |         //初始化导入sheet页实体类型下标
 | 
						
						
						
							|  |  |         int objListClass = 0;
 | 
						
						
						
							|  |  |         for (int i = 0; i < index; i++) {
 | 
						
						
						
							|  |  |             if (indexList.contains(i)) {
 | 
						
						
						
							|  |  |                 UploadDataListener<T> uploadDataListener = new UploadDataListener<>();
 | 
						
						
						
							|  |  |                 List<T> excels;
 | 
						
						
						
							|  |  |                 EasyExcelFactory.read(multipartFile.getInputStream(), objList.get(objListClass).getClass(), uploadDataListener).sheet(i).doRead();
 | 
						
						
						
							|  |  |                 excels = uploadDataListener.getList();
 | 
						
						
						
							|  |  |                 resultList.add(excels);
 | 
						
						
						
							|  |  |                 objListClass++;
 | 
						
						
						
							|  |  |             }
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |         return resultList;
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 读取多个sheet
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param file:文件流
 | 
						
						
						
							|  |  |      * @param index:需要读取的sheet个数                              [默认0开始,如果传入3,则读取0 1 2]
 | 
						
						
						
							|  |  |      * @param params:每个sheet里面需要封装的对象[如果index为3,则需要传入对应的3个对象]
 | 
						
						
						
							|  |  |      * @param <T>
 | 
						
						
						
							|  |  |      * @return
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> List<List<T>> importExcels(MultipartFile file, int index, List<Object> params) throws Exception {
 | 
						
						
						
							|  |  |         this.checkFile(file);
 | 
						
						
						
							|  |  |         List<List<T>> resultList = new LinkedList<>();
 | 
						
						
						
							|  |  |         for (int i = 0; i < index; i++) {
 | 
						
						
						
							|  |  |             UploadDataListener<T> uploadDataListener = new UploadDataListener<>();
 | 
						
						
						
							|  |  |             ExcelReaderBuilder builder = EasyExcelFactory.read(file.getInputStream(), params.get(i).getClass(), uploadDataListener);
 | 
						
						
						
							|  |  |             builder.sheet(i).doRead();
 | 
						
						
						
							|  |  |             List<T> list = uploadDataListener.getList();
 | 
						
						
						
							|  |  |             resultList.add(list);
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |         return resultList;
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 导出excel表格
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param response          :
 | 
						
						
						
							|  |  |      * @param dataList          :数据列表
 | 
						
						
						
							|  |  |      * @param clazz             :数据对象
 | 
						
						
						
							|  |  |      * @param fileName          :文件名称
 | 
						
						
						
							|  |  |      * @param sheetName:sheet名称
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> void exportExcel(HttpServletResponse response, List<T> dataList, Class<T> clazz, String fileName, String sheetName) throws Exception {
 | 
						
						
						
							|  |  |         response.setContentType("application/vnd.ms-excel");
 | 
						
						
						
							|  |  |         response.setCharacterEncoding(StandardCharsets.UTF_8.name());
 | 
						
						
						
							|  |  |         fileName = URLEncoder.encode(fileName, StandardCharsets.UTF_8.name());
 | 
						
						
						
							|  |  |         response.setHeader("Content-disposition", "attachment;filename=" + fileName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc());
 | 
						
						
						
							|  |  |         ServletOutputStream outputStream = response.getOutputStream();
 | 
						
						
						
							|  |  |         EasyExcelFactory.write(outputStream, clazz).sheet(sheetName).doWrite(dataList);
 | 
						
						
						
							|  |  |         outputStream.flush();
 | 
						
						
						
							|  |  |         outputStream.close();
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 导出多个sheet
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param response:
 | 
						
						
						
							|  |  |      * @param dataList:多个数据列表
 | 
						
						
						
							|  |  |      * @param clazzMap:对应每个列表里面的数据对应的sheet名称
 | 
						
						
						
							|  |  |      * @param fileName:文件名
 | 
						
						
						
							|  |  |      * @param <T>
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> void exportExcels(HttpServletResponse response, List<List<?>> dataList, Map<Integer, String> clazzMap, String fileName) throws Exception {
 | 
						
						
						
							|  |  |         response.setContentType("application/vnd.ms-excel");
 | 
						
						
						
							|  |  |         response.setCharacterEncoding(StandardCharsets.UTF_8.name());
 | 
						
						
						
							|  |  |         fileName = URLEncoder.encode(fileName, StandardCharsets.UTF_8.name());
 | 
						
						
						
							|  |  |         response.setHeader("Content-disposition", "attachment;filename=" + fileName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc());
 | 
						
						
						
							|  |  |         ExcelWriter excelWriter = EasyExcel.write(response.getOutputStream()).build();
 | 
						
						
						
							|  |  |         int len = dataList.get(0).size();
 | 
						
						
						
							|  |  |         for (int i = 0; i < len; i++) {
 | 
						
						
						
							|  |  |             List<?> objects = (List<?>) dataList.get(0).get(i);
 | 
						
						
						
							|  |  |             Class<?> aClass = objects.get(0).getClass();
 | 
						
						
						
							|  |  |             WriteSheet writeSheet0 = EasyExcel.writerSheet(i, clazzMap.get(i)).head(aClass).build();
 | 
						
						
						
							|  |  |             excelWriter.write(objects, writeSheet0);
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |         excelWriter.finish();
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 根据模板将集合对象填充表格-单个sheet
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param list:填充对象集合
 | 
						
						
						
							|  |  |      * @param object            :填充对象
 | 
						
						
						
							|  |  |      * @param fileName:文件名称
 | 
						
						
						
							|  |  |      * @param templateName:模板名称
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> void exportTemplateExcels(HttpServletResponse response, List<T> list, Object object, String fileName, String templateName) throws Exception {
 | 
						
						
						
							|  |  |         String template = ExcelTemplateEnum.TEMPLATE_PATH.getDesc() + File.separator + templateName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc();
 | 
						
						
						
							|  |  |         InputStream inputStream = this.getClass().getClassLoader().getResourceAsStream(template);
 | 
						
						
						
							|  |  |         FillConfig fillConfig = FillConfig.builder().forceNewRow(Boolean.TRUE).build();
 | 
						
						
						
							|  |  |         ExcelWriter excelWriter = EasyExcelFactory.write(getOutputStream(fileName, response)).withTemplate(inputStream).build();
 | 
						
						
						
							|  |  |         WriteSheet writeSheet0 = EasyExcelFactory.writerSheet(0).build();
 | 
						
						
						
							|  |  |         excelWriter.fill(object, fillConfig, writeSheet0);
 | 
						
						
						
							|  |  |         excelWriter.fill(list, fillConfig, writeSheet0);
 | 
						
						
						
							|  |  |         excelWriter.finish();
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 根据模板将集合对象填充表格-多个sheet
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param list1:填充对象集合
 | 
						
						
						
							|  |  |      * @param list2:填充对象集合
 | 
						
						
						
							|  |  |      * @param object1           :填充对象
 | 
						
						
						
							|  |  |      * @param object2           :填充对象
 | 
						
						
						
							|  |  |      * @param fileName:文件名称
 | 
						
						
						
							|  |  |      * @param templateName:模板名称
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> void exportSheetTemplateExcels(HttpServletResponse response, List<T> list1, List<T> list2, Object object1, Object object2, String fileName, String templateName) throws Exception {
 | 
						
						
						
							|  |  |         String template = ExcelTemplateEnum.TEMPLATE_PATH.getDesc() + File.separator + templateName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc();
 | 
						
						
						
							|  |  |         InputStream inputStream = this.getClass().getClassLoader().getResourceAsStream(template);
 | 
						
						
						
							|  |  |         FillConfig fillConfig = FillConfig.builder().forceNewRow(Boolean.TRUE).build();
 | 
						
						
						
							|  |  |         ExcelWriter excelWriter = EasyExcelFactory.write(getOutputStream(fileName, response)).withTemplate(inputStream).build();
 | 
						
						
						
							|  |  |         WriteSheet writeSheet0 = EasyExcelFactory.writerSheet(0).build();
 | 
						
						
						
							|  |  |         WriteSheet writeSheet1 = EasyExcelFactory.writerSheet(1).build();
 | 
						
						
						
							|  |  |         excelWriter.fill(object1, fillConfig, writeSheet0);
 | 
						
						
						
							|  |  |         excelWriter.fill(list1, fillConfig, writeSheet0);
 | 
						
						
						
							|  |  |         excelWriter.fill(object2, fillConfig, writeSheet1);
 | 
						
						
						
							|  |  |         excelWriter.fill(list2, fillConfig, writeSheet1);
 | 
						
						
						
							|  |  |         excelWriter.finish();
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 根据模板将单个对象填充表格
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param object            :填充对象
 | 
						
						
						
							|  |  |      * @param templateName:模板名称
 | 
						
						
						
							|  |  |      * @param fileName          :文件名称
 | 
						
						
						
							|  |  |      * @param sheetName         :需要写入的sheet名称 [不传:填充到第一个sheet]
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public void exportTemplateExcel(HttpServletResponse response, Object object, String templateName, String fileName, String sheetName) throws Exception {
 | 
						
						
						
							|  |  |         String template = ExcelTemplateEnum.TEMPLATE_PATH.getDesc() + File.separator + templateName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc();
 | 
						
						
						
							|  |  |         InputStream inputStream = this.getClass().getClassLoader().getResourceAsStream(template);
 | 
						
						
						
							|  |  |         if (StringUtils.isEmpty(sheetName)) {
 | 
						
						
						
							|  |  |             EasyExcelFactory.write(getOutputStream(fileName, response)).withTemplate(inputStream).sheet().doFill(object);
 | 
						
						
						
							|  |  |         } else {
 | 
						
						
						
							|  |  |             EasyExcelFactory.write(getOutputStream(fileName, response)).withTemplate(inputStream).sheet(sheetName).doFill(object);
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 根据模板将集合对象填充表格
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param list:填充对象集合
 | 
						
						
						
							|  |  |      * @param fileName:文件名称
 | 
						
						
						
							|  |  |      * @param templateName:模板名称
 | 
						
						
						
							|  |  |      * @param sheetName:需要写入的sheet [不传:填充到第一个sheet]
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> void exportTemplateExcelList(HttpServletResponse response, List<T> list, String fileName, String templateName, String sheetName) throws Exception {
 | 
						
						
						
							|  |  |         log.info("模板名称:{}", templateName);
 | 
						
						
						
							|  |  |         String template = ExcelTemplateEnum.TEMPLATE_PATH.getDesc() + File.separator + templateName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc();
 | 
						
						
						
							|  |  |         log.info("模板路径:{}", template);
 | 
						
						
						
							|  |  |         InputStream inputStream = this.getClass().getClassLoader().getResourceAsStream(template);
 | 
						
						
						
							|  |  |         // 全部填充:全部加载到内存中一次填充
 | 
						
						
						
							|  |  |         if (StringUtils.isEmpty(sheetName)) {
 | 
						
						
						
							|  |  |             EasyExcelFactory.write(getOutputStream(fileName, response)).withTemplate(inputStream).sheet().doFill(list);
 | 
						
						
						
							|  |  |         } else {
 | 
						
						
						
							|  |  |             EasyExcelFactory.write(getOutputStream(fileName, response)).withTemplate(inputStream).sheet(sheetName).doFill(list);
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 根据模板将集合对象填充表格
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param list:填充对象集合
 | 
						
						
						
							|  |  |      * @param fileName:文件名称
 | 
						
						
						
							|  |  |      * @param templateName:模板名称
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     public <T> void exportTemplateExcel2(HttpServletResponse response, List<T> list, String fileName, String templateName) throws Exception {
 | 
						
						
						
							|  |  |         String template = ExcelTemplateEnum.TEMPLATE_PATH.getDesc() + File.separator + templateName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc();
 | 
						
						
						
							|  |  |         InputStream inputStream = this.getClass().getClassLoader().getResourceAsStream(template);
 | 
						
						
						
							|  |  |         ExcelWriter excelWriter = EasyExcelFactory.write(getOutputStream(fileName, response)).withTemplate(inputStream).build();
 | 
						
						
						
							|  |  |         WriteSheet writeSheet = EasyExcelFactory.writerSheet().build();
 | 
						
						
						
							|  |  |         excelWriter.fill(list, writeSheet);
 | 
						
						
						
							|  |  |         excelWriter.finish();
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 构建输出流
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param fileName:文件名称
 | 
						
						
						
							|  |  |      * @param response:
 | 
						
						
						
							|  |  |      * @return
 | 
						
						
						
							|  |  |      * @throws Exception
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     private OutputStream getOutputStream(String fileName, HttpServletResponse response) throws Exception {
 | 
						
						
						
							|  |  |         fileName = URLEncoder.encode(fileName, StandardCharsets.UTF_8.name());
 | 
						
						
						
							|  |  |         response.setContentType("application/vnd.ms-excel");
 | 
						
						
						
							|  |  |         response.setCharacterEncoding(StandardCharsets.UTF_8.name());
 | 
						
						
						
							|  |  |         response.setHeader("Content-Disposition", "attachment;filename=" + fileName + ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc());
 | 
						
						
						
							|  |  |         return response.getOutputStream();
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  |     /**
 | 
						
						
						
							|  |  |      * 文件格式校验
 | 
						
						
						
							|  |  |      *
 | 
						
						
						
							|  |  |      * @param file:
 | 
						
						
						
							|  |  |      */
 | 
						
						
						
							|  |  |     private void checkFile(MultipartFile file) {
 | 
						
						
						
							|  |  |         if (file == null) {
 | 
						
						
						
							|  |  |             throw new RenException("文件不能为空");
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |         String fileName = file.getOriginalFilename();
 | 
						
						
						
							|  |  |         if (StringUtils.isEmpty(fileName)) {
 | 
						
						
						
							|  |  |             throw new RenException("文件不能为空");
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |         if (!fileName.endsWith(ExcelTemplateEnum.TEMPLATE_SUFFIX.getDesc())
 | 
						
						
						
							|  |  |                 && !fileName.endsWith(ExcelTemplateEnum.TEMPLATE_SUFFIX_XLS.getDesc())) {
 | 
						
						
						
							|  |  |             throw new RenException("请上传.xlsx或.xls文件");
 | 
						
						
						
							|  |  |         }
 | 
						
						
						
							|  |  |     }
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | 
 | 
						
						
						
							|  |  | }
 |