java POI导入导出Excel数据

主要的难点在service层解析excel表格数据这里,因为java这里有好多种数据,一旦excel表格里面不是java对应的数据类型就会报错。还有就是在数据库批量导入的时候mybatis有大小限制的

首先是导入前的下载模板

public class HolidayConfigController {
@GetMapping(value = "/downloadTemplate")
    public void downloadTemplate(HttpServletResponse response, HttpServletRequest request){
        log.info("HolidayController.downloadTemplate success Enter this method");
        try {
            //生成Excel文件
            HSSFWorkbook hssfWorkbook = new HSSFWorkbook();
            HSSFSheet sheet = hssfWorkbook.createSheet("Sheet0");
            // 设置表格默认列宽度为15个字节
            sheet.setDefaultColumnWidth((short)15);
            HSSFRow headRow = sheet.createRow(0);
            headRow.createCell(0).setCellValue("节假日日期");
            headRow.createCell(1).setCellValue("节假日描述");
            HSSFRow dataRow = sheet.createRow(sheet.getLastRowNum()+1);
            dataRow.createCell(0).setCellValue("2018-10-01");
            dataRow.createCell(1).setCellValue("国庆节");
            dataRow.createCell(2).setCellValue("请严格按此格式录入数据");
            response.setContentType("application/vnd.ms-excel;charset=UTF-8");
            String filename = "节假日数据导入模板.xls";
            String agent = request.getHeader("user-agent");//获得游览器
            String downloadFilename = FileUtil.encodeDownloadFilename(filename, agent); //使用工具类解决文件乱码的问题
            response.addHeader("Content-Disposition", "attachment;filename="+downloadFilename);
            ServletOutputStream outputStream = response.getOutputStream();
            hssfWorkbook.write(outputStream);
            //关闭
            hssfWorkbook.close();
        } catch (IOException e) {
            log.error("HolidayController.downloadTemplate Exception",e);
        }
    }

使用FileUtil工具类处理因浏览器不同导致的乱码问题

public class FileUtil {
    /**
     * 下载文件时,针对不同浏览器,进行附件名的编码
     *
     * @param filename
     *            下载文件名
     * @param agent
     *            客户端浏览器
     * @return 编码后的下载附件名
     * @throws IOException
     */
    public static String encodeDownloadFilename(String filename, String agent)
            throws IOException {
        if (agent.contains("Firefox")) { // 火狐浏览器
            filename = "=?UTF-8?B?"
                    + new BASE64Encoder().encode(filename.getBytes("utf-8"))
                    + "?=";
            filename = filename.replaceAll("\r\n", "");
        } else { // IE及其他浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+"," ");
        }
        return filename;
    }
}

其次是导入Execl数据,这里分2003版和2007版两个版本,同时用一种解析方式解析

@PostMapping(value = "/importExecl")
    public PojoResult importExecl(@RequestParam("file")MultipartFile file){
        try{
            InputStream inputStream = file.getInputStream();
            if(!inputStream.markSupported()) {
                inputStream = new PushbackInputStream(inputStream, 8);
            }
            if(POIFSFileSystem.hasPOIFSHeader(inputStream)) {
                log.info("HolidayController.importExecl Execl is 2003 and below");
                Workbook workbook = new HSSFWorkbook(new POIFSFileSystem(file.getInputStream()));
                String importXls = HolidayConfigService .importXlsOrXlsx(workbook);
                if("success".equals(importXls)){
                    log.info("HolidayController.importExecl Execl data import successfully");
                    return PojoResultUtil.success(Boolean.TRUE);
                }
                log.error("HolidayController.importExecl Execl data import failed");
                ProductionMessageCode.BATCH_IMPORT_FAILURE.setMessage(importXls);
                return PojoResultUtil.fail(ProductionMessageCode.BATCH_IMPORT_FAILURE);
            }else if(POIXMLDocument.hasOOXMLHeader(inputStream)){
                log.info("HolidayController.importExecl Execl is version 2007 and above");
                Workbook  workbook = new XSSFWorkbook(file.getInputStream());
                String importXls = HolidayConfigService .importXlsOrXlsx(workbook);
                if("success".equals(importXls)){
                    log.info("HolidayController.importExecl Execl data import successfully");
                    return PojoResultUtil.success(Boolean.TRUE);
                }
                log.error("HolidayController.importExecl Execl data import failed");
                ProductionMessageCode.BATCH_IMPORT_FAILURE.setMessage(importXls);
                return PojoResultUtil.fail(ProductionMessageCode.BATCH_IMPORT_FAILURE);
            }else {
                log.error("HolidayController.importExecl Execl Format does not exist");
                return PojoResultUtil.fail(ProductionMessageCode.EXECL_NOT_EXISTS);
            }
        }catch (Exception e){
            log.error("HolidayController.importExecl Execl data import Execption",e);
            return PojoResultUtil.fail(ProductionMessageCode.QUERY_FAIL);
        }
}

然后是service层的解析方式

public class HolidayConfigService {
    @Transactional
    public String importXlsOrXlsx(Workbook workbook){
        UserInfo userInfo = UserInfoThreadLocal.get();
        //获取Excel第一个sheet页
        DataFormatter formatter = new DataFormatter();
        List<HolidayConfigDO> holidayConfigs = new ArrayList<>();
        //定义错误信息记录集合
        List<String> list = new ArrayList<>();
        for(int numOfSheet = 0;numOfSheet < workbook.getNumberOfSheets(); numOfSheet++){
            Sheet sheets =workbook.getSheetAt(numOfSheet);
            list.add("第Sheet" + numOfSheet +"页");
            //遍历行row
            for(int rowNum = 0; rowNum<=sheets.getLastRowNum();rowNum++){
                Row row = sheets.getRow(rowNum);
                if(row.getRowNum()==0){
                    //如果是第一行直接跳出本次循环
                    continue;
                }
                HolidayConfigDO holidayConfigDO = new HolidayConfigDO();
                //将每一行数据的每一单元格数据取出
                try {
                    SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
                    String dateCellValue = dateFormat.format(row.getCell(0).getDateCellValue());
                    if(StringUtils.isEmpty(dateCellValue)){
                        list.add("第"+(rowNum+1)+"行,第1格节假日日期不能为空");
                    }else{
                        Date day = dateFormat.parse(dateCellValue);
                        holidayConfigDO.setDay(day);
                    }
                } catch (Exception e) {
                    list.add("第"+(rowNum+1)+"行,第1格节假日日期格式错误");
                }
                String description = formatter.formatCellValue(row.getCell(1));
                if(StringUtils.isEmpty(description)){
                    list.add("第"+(rowNum+1)+"行,第2格节假日描述不能为空");
                }else{
                    holidayConfigDO.setDescription(description);
                }
                holidayConfigDO.setDeleted(false);
                holidayConfigDO.setType((byte)1);
                holidayConfigDO.setCreator(userInfo.getEmpId());
                holidayConfigDO.setModifier(userInfo.getEmpId());
                holidayConfigs.add(holidayConfigDO);
            }
            //如果没有错误就将错误记录集合数据清空
            if(!CollectionUtils.isEmpty(list) && list.size() == 1){
                list.remove(0);
            }
        }
        if(!CollectionUtils.isEmpty(list)){
            return list.toString();
        }
        holidayConfigManager.batchInsertHolidayConfig(holidayConfigs);
        return "success";
    }

在附上批量插入数据的Mapper.xml文件

<insert id="batchInsertHolidayConfig">
        insert into dcc_holiday_config
        (gmt_create, gmt_modified, creator, modifier,day,type,is_deleted,description)
        values
        <foreach collection ="holidayConfigs" item="holidayConfig" index= "index" separator =",">
            (
            current_timestamp,
            current_timestamp,
            #{holidayConfig.creator,jdbcType=VARCHAR},
            #{holidayConfig.modifier,jdbcType=VARCHAR},
            #{holidayConfig.day,jdbcType=DATE},
            #{holidayConfig.type,jdbcType=TINYINT},
            #{holidayConfig.deleted,jdbcType=BIT},
            #{holidayConfig.description,jdbcType=VARCHAR}
            )
        </foreach >
        ON DUPLICATE KEY UPDATE
        gmt_create = values (gmt_create),
        gmt_modified = values (gmt_modified),
        creator = values (creator),
        modifier = values (modifier),
        day = values (day),
        type = values (type),
        is_deleted = values (is_deleted),
        description = values (description)
    </insert>

使用EasyPoi导出

引入相关maven依赖

<dependency>
    <groupId>cn.afterturn</groupId>
    <artifactId>easypoi-base</artifactId>
    <version>4.1.3</version>
</dependency>

<dependency>
    <groupId>cn.afterturn</groupId>
    <artifactId>easypoi-annotation</artifactId>
    <version>4.1.3</version>
</dependency>

<dependency>
    <groupId>cn.afterturn</groupId>
    <artifactId>easypoi-web</artifactId>
    <version>4.1.3</version>
</dependency>

EasyPOI 导出代码示例

ExportParams exportParams = new ExportParams();
exportParams.setStyle(ExcelExportStyler.class); // 设置样式
Workbook workbook = ExcelExportUtil.exportExcel(exportParams, CallPoolExcelVo.class, voList);

其中 ExcelExportStyler 就是我的自定义样式类。

官方默认提供了一个样式枚局类 ExcelStyleType,但是在 4.2 的版本中,BORDER、COLOR 这两个已经被标记为淘汰,不建议继续使用。

官方地址介绍:https://opensource.afterturn.cn/doc/easypoi.html#304

public enum ExcelStyleType {
    NONE("默认样式", ExcelExportStylerDefaultImpl.class),
    BORDER("边框样式", ExcelExportStylerBorderImpl.class),
    COLOR("间隔行样式", ExcelExportStylerColorImpl.class);
    
    ....
}

所以,我们可以继承 ExcelExportStylerDefaultImpl 或者实现 IExcelExportStyler 接口,完善其中方法即可。

public interface IExcelExportStyler {
    /**
     * 列表头样式
     * @param headerColor
     * @return
     */
    public CellStyle getHeaderStyle(short headerColor);
    /**
     * 标题样式
     * @param color
     * @return
     */
    public CellStyle getTitleStyle(short color);
    /**
     * 获取样式方法
     * @param Parity
     * @param entity
     * @return
     */
    public CellStyle getStyles(boolean Parity, ExcelExportEntity entity);
}

创建自定义导出Excel样式类

/**
 * @author: huangyibo
 * @Date: 2020/11/10 16:59
 * @Description: easypoi 导出excel样式修改
 */
public class ExcelStyleUtil extends AbstractExcelExportStyler implements IExcelExportStyler {
    private static final short STRING_FORMAT = (short) BuiltinFormats.getBuiltinFormat("TEXT");
    private static final short FONT_SIZE_TEN = 10;
    private static final short FONT_SIZE_ELEVEN = 18;
    private static final short FONT_SIZE_TWELVE = 18;
    private static final short FONT_Height_LEN = 10;

    /**
     * 大标题样式
     */
    private CellStyle headerStyle;
    /**
     * 每列标题样式
     */
    private CellStyle titleStyle;
    /**
     * 数据行样式
     */
    private CellStyle styles;

    public ExcelStyleUtil(Workbook workbook) {
        this.init(workbook);
    }

    /**
     * 初始化样式
     *
     * @param workbook
     */
    private void init(Workbook workbook) {
        this.headerStyle = initHeaderStyle(workbook);
        this.titleStyle = initTitleStyle(workbook);
        this.styles = initStyles(workbook);
    }

    /**
     * 大标题样式
     *
     * @param color
     * @return
     */
    @Override
    public CellStyle getHeaderStyle(short color) {
        return headerStyle;
    }

    /**
     * 每列标题样式
     *
     * @param color
     * @return
     */
    @Override
    public CellStyle getTitleStyle(short color) {
        return titleStyle;
    }

    /**
     * 数据行样式
     *
     * @param parity 可以用来表示奇偶行
     * @param entity 数据内容
     * @return 样式
     */
    @Override
    public CellStyle getStyles(boolean parity, ExcelExportEntity entity) {
        return styles;
    }

    /**
     * 获取样式方法
     *
     * @param dataRow 数据行
     * @param obj     对象
     * @param data    数据
     */
    @Override
    public CellStyle getStyles(Cell cell, int dataRow, ExcelExportEntity entity, Object obj, Object data) {
        return getStyles(true, entity);
    }

    /**
     * 模板使用的样式设置
     */
    @Override
    public CellStyle getTemplateStyles(boolean isSingle, ExcelForEachParams excelForEachParams) {
        return null;
    }

    /**
     * 初始化--大标题样式
     *
     * @param workbook
     * @return
     */
    private CellStyle initHeaderStyle(Workbook workbook) {
        CellStyle style = getBaseCellStyle(workbook);
        style.setFont(getFont(workbook, FONT_SIZE_TWELVE, true));
        return style;
    }

    /**
     * 初始化--每列标题样式
     *
     * @param workbook
     * @return
     */
    private CellStyle initTitleStyle(Workbook workbook) {
        CellStyle style = getBaseCellStyle(workbook);
        style.setFont(getFont(workbook, FONT_SIZE_ELEVEN, false));
        //背景色
        style.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.getIndex());
        style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
        return style;
    }

    /**
     * 初始化--数据行样式
     *
     * @param workbook
     * @return
     */
    private CellStyle initStyles(Workbook workbook) {
        CellStyle style = getBaseCellStyle(workbook);
        style.setFont(getFont(workbook, FONT_SIZE_TEN, false));
        style.setDataFormat(STRING_FORMAT);
        return style;
    }

    /**
     * 基础样式
     *
     * @return
     */
    private CellStyle getBaseCellStyle(Workbook workbook) {
        CellStyle style = workbook.createCellStyle();

        //水平居中
        style.setAlignment(HorizontalAlignment.CENTER);
        //上下居中
        style.setVerticalAlignment(VerticalAlignment.CENTER);
        //设置自动换行
        style.setWrapText(true);
        return style;
    }

    /**
     * 字体样式
     *
     * @param size   字体大小
     * @param isBold 是否加粗
     * @return
     */
    private Font getFont(Workbook workbook, short size, boolean isBold) {
        Font font = workbook.createFont();
        //字体样式
        font.setFontName("宋体");
        //是否加粗
        font.setBold(isBold);
        //字体大小
        font.setFontHeightInPoints(size);
        return font;
    }
}

创建核心导出工具类

/**
 * @author: huangyibo
 * @Date: 2020/11/10 16:57
 * @Description:
 */
public class EasyPOIUtil {

    /**
     * 下载文件时,针对不同浏览器,进行附件名的编码
     *
     * @param filename
     *            下载文件名
     * @param agent
     *            客户端浏览器
     * @return 编码后的下载附件名
     * @throws IOException
     */
    public static String encodeDownloadFilename(String filename, String agent)
            throws IOException {
        if (agent.contains("Firefox")) { // 火狐浏览器
            filename = "=?UTF-8?B?"
                    + Base64.getEncoder().encode(filename.getBytes("utf-8"))
                    + "?=";
            filename = filename.replaceAll("\r\n", "");
        } else { // IE及其他浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+"," ");
        }
        return filename;
    }

    /**
     * 设置导出Excel格式
     * @param response
     * @param request
     * @param filename
     * @throws IOException
     */
    public static void setExportExcelFormat(HttpServletResponse response, HttpServletRequest request,String filename) throws IOException {
        String agent = request.getHeader("user-agent");//获得游览器
        filename = filename + ".xls";
        String downloadFilename = encodeDownloadFilename(filename, agent); //使用工具类解决文件乱码的问题
        response.setCharacterEncoding("UTF-8");
        // 设置响应输出的头类型
        response.setHeader("content-Type", "application/vnd.ms-excel");
        // 下载文件的默认名称
        response.setHeader("Content-Disposition", "attachment;filename="+downloadFilename);
        response.setHeader("Pragma", "public");
        response.setHeader("Cache-Control", "no-store");
        response.addHeader("Cache-Control", "max-age=0");
    }

    public static void writeExcel(HttpServletResponse response, HttpServletRequest request, Class<?> clazz, List<?> exportModelList, String filename, String sheetName) throws IOException {
    
        //设置导出Excel格式
        setExportExcelFormat(response,request,filename);

        ExportParams exportParams = new ExportParams();
        exportParams.setStyle(ExcelStyleUtil.class); // 设置样式
        // 设置sheet得名称
        exportParams.setSheetName(sheetName);

        Map<String, Object> deptExportMap = new HashMap<>();
        // title的参数为ExportParams类型,目前仅仅在ExportParams中设置了sheetName
        deptExportMap.put("title", exportParams);
        // 模版导出对应得实体类型
        deptExportMap.put("entity", clazz);
        // sheet中要填充得数据
        deptExportMap.put("data", exportModelList);

        List<Map<String, Object>> sheetsList = new ArrayList<>();
        sheetsList.add(deptExportMap);
        Workbook workbook = ExcelExportUtil.exportExcel(sheetsList, ExcelType.HSSF);
        OutputStream outputStream = response.getOutputStream();
        workbook.write(outputStream);
        outputStream.flush();
        outputStream.close();
    }
}

创建导出对象实体Bean

关于合并都在实体类里面(主要是结构和注解)

关于合并只有两点:
1、类的数据结构。
2、注解
    @ExcelCollection(name = "") 集合使用
    @Excel(name = "采购数量合计(支)", width = 30, orderNum = "5",needMerge = true, isStatistics = true)
        name:列的标题名称,width :表格的宽度, orderNum :排序(注意每个类中的排序为单独的  不同类排序不通用) 
        needMerge : 是否合并单元格, isStatistics:是否进行统计(若为true,则在表格最后自动生成最终统计)

@Data
@ExcelTarget("empUtil")
public class DeptUtil {

    @Excel(name = "部门编号", width = 30 , needMerge = true)
    private Integer id;

    @Excel(name = "部门名称", width = 30 , needMerge = true)
    private String deptName;

    @Excel(name = "薪酬", width = 30, type = 10,replace="{0_null}")
    private BigDecimal salary;

    @ExcelCollection(name = "")
    private List<EmpUtil> emps;

}

@Data
@ExcelTarget("empUtil")
public class EmpUtil {

    @Excel(name = "序号", width = 30, groupName="你好")
    private Integer id;

    @Excel(name = "员工姓名", width = 30,groupName="你好")
    private String empName;

    @Excel(name = "年龄", width = 30, type = 10)
    private Integer age;

    @Excel(name = "入职时间", width = 30, format = "yyyy-MM-dd")
    private Date hiredate;

    @Excel(name = "薪酬", width = 30, type = 10,replace="{0_null}")
    private BigDecimal salary;
}

导出controller

@RestController
@RequestMapping("/export")
@Slf4j
public class ExportController {

    @GetMapping("/excel")
    public String excel(HttpServletResponse response, HttpServletRequest request){
        try {
            List<DeptUtil> exportList = selectDeptList();
            String fileName = "员工管理";
            String sheetName = "sheet1";
            EasyPOIUtil.writeExcel(response, request, DeptUtil.class,exportList, fileName, sheetName);
        } catch (Exception e) {
            log.error("error", e);
        }
        return "success";
    }
}

多sheet页导入导出

public class EasyPoiUtil {

    /**
     * 下载文件时,针对不同浏览器,进行附件名的编码
     *
     * @param filename
     *            下载文件名
     * @param agent
     *            客户端浏览器
     * @return 编码后的下载附件名
     * @throws IOException
     */
    public static String encodeDownloadFilename(String filename, String agent)
            throws IOException {
        if (agent.contains("Firefox")) { // 火狐浏览器
            filename = "=?UTF-8?B?"
                    + Base64.getEncoder().encode(filename.getBytes("utf-8"))
                    + "?=";
            filename = filename.replaceAll("\r\n", "");
        } else { // IE及其他浏览器
            filename = URLEncoder.encode(filename, "utf-8");
            filename = filename.replace("+"," ");
        }
        return filename;
    }

    /**
     * 设置导出Excel格式
     * @param response
     * @param request
     * @param filename
     * @throws IOException
     */
    public static void setExportExcelFormat(HttpServletResponse response, HttpServletRequest request, String filename) throws IOException {
        String agent = request.getHeader("user-agent");//获得游览器
        filename = filename + ".xls";
        String downloadFilename = encodeDownloadFilename(filename, agent); //使用工具类解决文件乱码的问题
        response.setCharacterEncoding("UTF-8");
        // 设置响应输出的头类型
        response.setHeader("content-Type", "application/vnd.ms-excel");
        // 下载文件的默认名称
        response.setHeader("Content-Disposition", "attachment;filename="+downloadFilename);
        response.setHeader("Pragma", "public");
        response.setHeader("Cache-Control", "no-store");
        response.addHeader("Cache-Control", "max-age=0");
    }

    /**
     * 构建 sheet
     * @param sheetName sheet 名字
     * @param clazz     clazz
     * @param list      列表
     * @return          {@link Map<String, Object>}
     */
    public static Map<String, Object> buildSheet(String sheetName, Class<?> clazz, List<?> list) {
        ExportParams exportParams = new ExportParams();
        exportParams.setType(ExcelType.XSSF);
        exportParams.setSheetName(sheetName);
        // 导出样式
        exportParams.setStyle(ExcelStyleUtil.class);

        Map<String, Object> map = new HashMap<>(8);
        // title的参数为ExportParams类型
        map.put("title", exportParams);
        // 模版导出对应得实体类型
        map.put("entity", clazz);
        // sheet中要填充得数据
        map.put("data", list);
        return map;
    }

    /**
     * 导出数据创建Workbook,便于转成MultipartFile,用于文件上传
     * Excel导出, 支持多sheet导出, 根据数据量自动切换sheet
     * Excel2003版最大行数是65536行。
     * Excel2007开始的版本最大行数是1048576行
     * Excel2010最大行数是1048576行
     * @param clazz
     * @param exportModelList
     * @return
     */
    public static Workbook createWorkbook(Class<?> clazz, List<?> exportModelList) {
        List<Map<String, Object>> sheetsList = new ArrayList<>();
        //按每65500条数据为一组分割
        List<? extends List<?>> partitionList = Lists.partition(exportModelList, 65500);
        for (int i = 0; i < partitionList.size(); i++) {
            Map<String, Object> exportMap = buildSheet("sheet" + i, clazz, partitionList.get(i));
            sheetsList.add(exportMap);
        }
        return ExcelExportUtil.exportExcel(sheetsList, ExcelType.HSSF);
    }

    /**
     * Excel导出, 支持多sheet导出, 根据数据量自动切换sheet
     * Excel2003版最大行数是65536行。
     * Excel2007开始的版本最大行数是1048576行
     * Excel2010最大行数是1048576行
     * @param response
     * @param request
     * @param clazz
     * @param exportModelList
     * @param filename
     * @throws IOException
     */
    public static void writeExcel(HttpServletResponse response, HttpServletRequest request, Class<?> clazz, List<?> exportModelList, String filename) throws IOException {
        //设置导出Excel格式
        setExportExcelFormat(response,request,filename);

        Workbook workbook = createWorkbook(clazz, exportModelList);
        OutputStream outputStream = response.getOutputStream();
        workbook.write(outputStream);
        outputStream.flush();
        outputStream.close();
    }

    /**
     * 得到 Workbook 对象
     * @param file
     * @return
     * @throws IOException
     */
    public static Workbook getWorkBook(MultipartFile file) throws IOException {
        //这样写  excel 能兼容03和07
        InputStream inputStream = file.getInputStream();
        Workbook hssfWorkbook = null;
        try {
            hssfWorkbook = new XSSFWorkbook(inputStream);
        } catch (Exception ex) {
            hssfWorkbook = new HSSFWorkbook(inputStream);
        }
        return hssfWorkbook;
    }

    /**
     * 导入Excel数据
     * @param file
     * @param sheetNum
     * @param titleRows
     * @param headerRows
     * @param pojoClass
     * @param <T>
     * @return
     */
    public static <T> ExcelImportResult<T> importExcel(MultipartFile file, Integer sheetNum, Integer titleRows, Integer headerRows, Class<T> pojoClass) {
        //导入的基本配置
        ImportParams params = new ImportParams();
        //sheet页索引
        params.setStartSheetIndex(sheetNum);
        //表头一行
        params.setHeadRows(headerRows);
        //标题一行
        params.setTitleRows(titleRows);
        //代表导入这里是需要验证的(根据字段上的注解校验)
        params.setNeedVerify(true);
        //设及一个自定义校验 (自定义校验名字不可重复)
        //params.setVerifyHandler(excelVerifyHandler);
        //使用框架自身导入工具
        ExcelImportResult<T> importResult = null;
        try {
            importResult = ExcelImportUtil.importExcelMore(file.getInputStream(), pojoClass, params);
        } catch (Exception e) {
            throw new RuntimeException(e.getMessage());
        }
        return importResult;
    }
}

导出

导出实体类

@Data
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@ApiModel(description="看房楼盘估价表")
public class DictGardenAppraisalExport implements Serializable {

    @ApiModelProperty(value = "城市")
    @Excel(name = "城市", orderNum = "0", width = 20)
    private String cityName;

    @ApiModelProperty(value = "区域")
    @Excel(name = "区域", orderNum = "1", width = 20)
    private String regionName;

    @ApiModelProperty(value = "商圈")
    @Excel(name = "商圈", orderNum = "2", width = 20)
    private String businessAreaName;

    @ApiModelProperty(value = "楼盘名称")
    @Excel(name = "楼盘名称", orderNum = "3", width = 20)
    private String gardenName;

    @ApiModelProperty(value = "物业类型(普通住宅/公寓/酒店)")
    @Excel(name = "物业类型", orderNum = "4", width = 20)
    private String usageName;

    @ApiModelProperty(value = "委托套数")
    @Excel(name = "委托套数", orderNum = "5", width = 20)
    private Integer trustNum;

    @ApiModelProperty(value = "委托均价")
    @Excel(name = "委托均价", orderNum = "6", width = 20)
    private BigDecimal trustAvgPrice;

    @ApiModelProperty(value = "委托总价")
    private BigDecimal trustTotalPrice;

    @ApiModelProperty(value = "委托总面积")
    private BigDecimal trustTotalArea;

    @ApiModelProperty(value = "成交套数")
    @Excel(name = "成交套数", orderNum = "7", width = 20)
    private Integer tradeNum;

    @ApiModelProperty(value = "成交均价")
    @Excel(name = "成交均价", orderNum = "8", width = 20)
    private BigDecimal tradeAvgPrice;

    @ApiModelProperty(value = "成交总价")
    private BigDecimal tradeTotalPrice;

    @ApiModelProperty(value = "成交总面积")
    private BigDecimal tradeTotalArea;

    @ApiModelProperty(value = "外部参考价")
    @Excel(name = "外部参考价", orderNum = "9", width = 20)
    private BigDecimal outerReferPrice;

    @ApiModelProperty(value = "参考价")
    @Excel(name = "参考价", orderNum = "10", width = 20)
    private BigDecimal referPrice;

    @ApiModelProperty(value = "政府指导价")
    @Excel(name = "政府指导价", orderNum = "11", width = 20)
    private BigDecimal governGuidePrice;

    @ApiModelProperty(value = "价差幅度1 (参考价/第三方委托均价)")
    @Excel(name = "价差幅度1", orderNum = "12", width = 20)
    private BigDecimal diffPriceRange1;

    @ApiModelProperty(value = "价差幅度2 (参考价/第三方成交均价)")
    @Excel(name = "价差幅度2", orderNum = "13", width = 20)
    private BigDecimal diffPriceRange2;

    @ApiModelProperty(value = "价差幅度3 (参考价/第三方参考价)")
    @Excel(name = "价差幅度3", orderNum = "14", width = 20)
    private BigDecimal diffPriceRange3;

    @ApiModelProperty(value = "估价日期")
    @Excel(name = "估价日期", exportFormat = "yyyy-MM-dd", orderNum = "15", width = 20)
    private Date referDate;

    @ApiModelProperty(value = "发布状态:已发布=PUBLISHED、待发布=TOBE_PUBLISH、已撤回=REVOKE")
    @Excel(name = "状态", orderNum = "16", width = 20)
    private String publishStatus;
}

导出逻辑

@PostMapping(value = "/exportGardenAppraisal")
@ApiOperation(value = "导出估价信息")
@ApiResponses({
        @ApiResponse(message = "导出估价信息", code = 200, response = DictGardenAppraisalVo.class)
})
public String exportGardenAppraisal(@RequestBody DictGardenAppraisalQueryPageForm queryPageForm, HttpServletResponse response, HttpServletRequest request) {
    queryPageForm.setPageSize(100000);
    Pagination<DictGardenAppraisalVo> page = dictGardenAppraisalRemote.queryDictGardenAppraisalPage(queryPageForm);
    if(page != null){
        if(!CollectionUtils.isEmpty(page.getItems())){
            List<DictGardenAppraisalExport> exportList = page.getItems().stream().map(item -> {
                DictGardenAppraisalExport export = new DictGardenAppraisalExport();
                BeanUtils.copyProperties(item, export);
                return export;
            }).collect(Collectors.toList());
            try {
                EasyPoiUtil.writeExcel(response, request, DictGardenAppraisalExport.class, exportList, "估价");
            } catch (IOException e) {
                log.error("估价数据导出excel Exception",e);
            }
        }
    }
    return successInfo(true);
}

导入

导入实体类

@Data
public class DictImportTradeDetailDTO implements Serializable {


    @ApiModelProperty(value = "批次号:yyyy-MM格式")
    @Excel(name = "日期")
    @NotNull(message = "日期不能为空")
    private String batchNum;

    @ApiModelProperty(value = "贝壳楼盘名")
    @Excel(name = "贝壳楼盘名")
    private String bkGardenName;

    @ApiModelProperty(value = "贝壳小区ID")
    @Excel(name = "贝壳小区ID")
    private String bkGardenId;

    @ApiModelProperty(value = "小区ID")
    @Excel(name = "小区ID")
    @NotNull(message = "小区ID不能为空")
    private String kfGardenId;

    @ApiModelProperty(value = "城市")
    @Excel(name = "城市")
    private String cityName;

    @ApiModelProperty(value = "区域")
    @Excel(name = "区域")
    private String regionName;

    @ApiModelProperty(value = "成交日期")
    @Excel(name = "成交时间", exportFormat = "yyyy-MM-dd", importFormat = "yyyy-MM-dd")
    private Date tradeDate;

    @ApiModelProperty(value = "房源价格")
    @Excel(name = "房源价格")
    @NotNull(message = "房源价格不能为空")
    @DecimalMin(value = "0.00",message = "房源价格不能小于0")
    private BigDecimal roomPrice;

    @ApiModelProperty(value = "房源面积(单位M2)")
    @Excel(name = "房源面积")
    @NotNull(message = "房源面积不能为空")
    @DecimalMin(value = "0.00",message = "房源面积不能小于0")
    private BigDecimal roomArea;

    @ApiModelProperty(value = "楼层")
    @Excel(name = "楼层")
    private String floor;

    @ApiModelProperty(value = "朝向")
    @Excel(name = "朝向")
    private String towards;

    @ApiModelProperty(value = "户型")
    @Excel(name = "户型")
    private String roomType;

    @ApiModelProperty(value = "房屋用途")
    @Excel(name = "房屋用途")
    private String roomUse;

    @ApiModelProperty(value = "房源链接")
    @Excel(name = "房源ID")
    @NotNull(message = "房源ID不能为空")
    private String roomLink;
}

导入具体逻辑

@RestController
@RequestMapping("/security/dictImportTradeDetail")
@Slf4j
public class DictImportTradeDetailController extends WebBaseController {

    @Resource
    private DictImportTradeDetailRemote dictImportTradeDetailRemote;

    @Resource
    private AliyunOss aliyunOss;

    @PostMapping("/import")
    @ApiOperation(value = "导入交易明细数据")
    @ApiResponses({
            @ApiResponse(message = "导入交易明细数据", code = 200, response = DictImportDataReturnVo.class)
    })
    public String importData(@RequestParam("file") MultipartFile file) {
        DictImportDataReturnVo returnVo = new DictImportDataReturnVo();
        try {
            JwtUserInfo userInfo = getCurrentUserInfo();
            Workbook workbook = EasyPoiUtil.getWorkBook(file);
            int sheetNum = workbook.getNumberOfSheets();

            List<DictImportTradeDetailDTO> failList = new ArrayList<>();
            List<DictImportTradeDetailDTO> successList = new ArrayList<>();
            for (int i = 0; i < sheetNum; i++) {
                ExcelImportResult<DictImportTradeDetailDTO> importResult = EasyPoiUtil.importExcel(file, i, 0, 1, DictImportTradeDetailDTO.class);
                //失败结果集
                List<DictImportTradeDetailDTO> failTradeList = importResult.getFailList();
                if(!CollectionUtils.isEmpty(failTradeList)){
                    failList.addAll(failTradeList);
                }
                //导入成功的数据
                List<DictImportTradeDetailDTO> successTradeList = importResult.getList();
                if(!CollectionUtils.isEmpty(successTradeList)){
                    successTradeList.forEach(trade -> {
                        //房源价格导入的单位为万,这里转为元
                        trade.setRoomPrice(trade.getRoomPrice().multiply(new BigDecimal(10000)));
                    });
                    successList.addAll(successTradeList);
                }
            }
            if(CollectionUtils.isEmpty(successList)){
                return successInfo(false);
            }
            DictImportDataRecordDTO dictImportDataRecordDTO = new DictImportDataRecordDTO();

            if(!CollectionUtils.isEmpty(failList)){
                //失败的数据需要写入Workbook
                Workbook workbookFile = EasyPoiUtil.createWorkbook(DictImportTradeDetailDTO.class, failList);
                //Workbook转成MultipartFile
                MultipartFile multipartFile = FileUtil.getMultipartFileFromWorkbook(workbookFile, "导入失败数据-"+file.getOriginalFilename());
                //上传aliyun 对象存储 OSS
                AliyunUploadRequest uploadRequest = new AliyunUploadRequest(multipartFile);
                AliyunUploadResult result = aliyunOss.fileUpload(uploadRequest);
                log.info("导入交易明细数据, 校验不通过数据写入excel上传到阿里云oss成功, result:{}", JSON.toJSONString(result));
                if(result != null){
                    dictImportDataRecordDTO.setFailDataFileUrl(result.getUrl());
                }
            }
            dictImportDataRecordDTO.setFailNum(failList.size());
            dictImportDataRecordDTO.setTradeDetailList(successList);
            dictImportDataRecordDTO.setImportTotalNum(successList.size() + failList.size());
            dictImportDataRecordDTO.setSuccessNum(successList.size());
            dictImportDataRecordDTO.setBatchNum(successList.get(0).getBatchNum());
            dictImportDataRecordDTO.setCityName(successList.get(0).getCityName());
            dictImportDataRecordDTO.setDataType(DataTypeEnum.TRADE_DATA);
            dictImportDataRecordDTO.setCreateId(userInfo.getUserId());
            dictImportDataRecordDTO.setModifyId(userInfo.getUserId());
            dictImportDataRecordDTO.setStatus(StatusEnum.ENABLED);
            dictImportTradeDetailRemote.importData(dictImportDataRecordDTO);
            returnVo.setImportTotalNum(dictImportDataRecordDTO.getImportTotalNum());
            returnVo.setSuccessNum(dictImportDataRecordDTO.getSuccessNum());
            returnVo.setFailNum(dictImportDataRecordDTO.getFailNum());
            returnVo.setSuccess(Boolean.TRUE);
            log.info("导入交易明细数据 SUCCESS, dictImportDataRecordDTO,{}, returnVo:{}", JSON.toJSONString(dictImportDataRecordDTO), JSON.toJSONString(returnVo));
        } catch (Exception e) {
            returnVo.setSuccess(Boolean.FALSE);
            returnVo.setErrCode(DictReturnCodeEnum.F6000000.name());
            returnVo.setErrMessage(DictReturnCodeEnum.F6000000.getMessage());
            log.info("导入交易明细数据 Exception, returnVo:{}", JSON.toJSONString(returnVo), e);
        }
        return successInfo(returnVo);
    }
}

参考:
http://www.ibloger.net/article/3392.html

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,608评论 18 399
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,646评论 18 139
  • 1、支持分批导入(可配置) 2、批量导入降低数据库操作 3、批量数据错误会自动重新生成导入数据最终只剩下错误数据,...
    隐形的稻草人_44ca阅读 668评论 0 0
  • 本文摘选自博客:Android权限操作之uses-permission详解 - Boblim - 博客园 本文实例...
    侧耳倾听ST阅读 1,894评论 0 1
  • 一直以来,所有的事情都会感觉良好,上学的时候感觉良好,但最终成为学渣,然后归于自己考试紧张,不适合理科种种。自己总...
    醉眼陶然阅读 393评论 0 0