easyexcel復雜表頭及標題的導出功能(自定義樣式及多sheet導出)


注意表格行高列寬統一設置是在實體類的類名注解上,如果需要對表格進行精細的寬高設置需要刪除掉這兩個注解,可以在攔截器使用row的方法進行設置

1、引入依賴

<dependency>
    <groupId>com.alibaba</groupId>
    <artifactId>easyexcel</artifactId>
     <version>2.1.4</version>
</dependency>

2、實體類(注解法)示例

@Data
@AllArgsConstructor
@NoArgsConstructor
@HeadRowHeight(20)
@ColumnWidth(25)
@HeadStyle(fillPatternType = FillPatternType.SOLID_FOREGROUND, fillForegroundColor = 22,horizontalAlignment = HorizontalAlignment.CENTER)
@ContentStyle(wrapped = true,verticalAlignment = VerticalAlignment.TOP)
@Builder
public class UserEntity implements Serializable {
    private static final long serialVersionUID = 1L;
    
    @ExcelProperty(value = "ID", order = 1)
    private Long id;
    
    @ExcelProperty(value = "姓名", order = 2)
    private String userName;
   
    @ExcelProperty(value = "年齡", order = 3)
    @ColumnWidth(15)
    private Integer age;
   
    @ExcelProperty(value = "郵箱", order = 4)
    private String email;
   
    @ExcelProperty(value = "出生日期", order = 5,converter = LocalDateConverter.class)
    private Date birth;
    
    @ExcelIgnore
    private String remark;
}
View Code
package com.jpxx.admin.pilebody.service.api.dto;
 
import com.alibaba.excel.annotation.ExcelIgnore;
import com.alibaba.excel.annotation.ExcelProperty;
import com.alibaba.excel.annotation.write.style.ColumnWidth;
import com.alibaba.excel.annotation.write.style.ContentRowHeight;
import com.alibaba.excel.annotation.write.style.HeadRowHeight;
import com.alibaba.excel.util.StringUtils;
import lombok.Data;
import lombok.NoArgsConstructor;
import lombok.experimental.Accessors;
 
 
@Data
@NoArgsConstructor
@Accessors(chain = true)
@ContentRowHeight(45)
@HeadRowHeight(50)
public class PilebodycheckMonthDto {
 
    @ExcelIgnore
    private String id;
 
    @ExcelIgnore
    private String cityid;
 
    @ExcelIgnore
    private String districtid;
 
    @ExcelProperty(value = {"序號","序號"},index = 0)
    @ColumnWidth(10)
    private String orderNum;
 
    @ExcelProperty(value = {"堆體名稱","堆體名稱"},index = 1)
    @ColumnWidth(15)
    private String name;
 
    @ExcelProperty(value = {"具體位置","具體位置"},index = 3)
    @ColumnWidth(30)
    private String address;
 
    @ExcelProperty(value = {"占地面積(平方)","占地面積(平方)"},index = 4)
    @ColumnWidth(15)
    private String areastr;
 
    @ExcelProperty(value = {"堆體高度(米)","堆體高度(米)"},index = 5)
    @ColumnWidth(10)
    private String heightstr;
 
    @ExcelProperty(value = {"建築垃圾堆存量(萬方)","建築垃圾堆存量(萬方)"},index = 6)
    @ColumnWidth(15)
    private String stocknum;
 
    @ExcelIgnore
    @Dict(dicCode = "governway")
    private String governway;
 
    @ExcelProperty(value = {"治理方式","治理方式"},index = 7)
    @ColumnWidth(20)
    private String governwayname;
 
    @ExcelProperty(value = {"如需外運,計划外運時間","如需外運,計划外運時間"},index = 8)
    @ColumnWidth(15)
    private String outwardtransporttime;
 
    @ExcelProperty(value = {"截止目前累計治理量(萬方)","截止目前累計治理量(萬方)"},index = 13)
    @ColumnWidth(15)
    private String governnum;
 
    @ExcelProperty(value = {"治理主體","治理主體"},index = 14)
    @ColumnWidth(15)
    private String governbody;
 
    @ExcelIgnore
    @Dict(dicCode = "typestr")
    private String typestr;
 
    @ExcelProperty(value = {"堆體類型","堆體類型"},index = 2)
    @ColumnWidth(15)
    private String typestrname;
 
    @ExcelIgnore
    @Dict(dicCode = "statestr")
    private String statestr;
 
    @ExcelIgnore
    private String districtname;
 
    @ExcelProperty(value = {"監管單位","監管單位"},index = 15)
    @ColumnWidth(15)
    private String supervisedepartname;
 
    @ExcelProperty(value = {"監管責任人","監管責任人"},index = 16)
    @ColumnWidth(10)
    private String supervisepeoname;
 
    @ExcelProperty(value = {"職務","職務"},index = 17)
    @ColumnWidth(10)
    private String supervisepeoposition;
 
    @ExcelProperty(value = {"聯系方式","聯系方式"},index = 18)
    @ColumnWidth(20)
    private String supervisepeophone;
 
    @ExcelIgnore
    private String residuenum;
 
    @ExcelIgnore
    private String governendtime;
 
    @ExcelIgnore
    private String governendyearmonth;
 
    @ExcelProperty(value = {"本月治理量(萬方)","外運量"},index = 9)
    @ColumnWidth(15)
    private String outwardtransportnum;
 
    @ExcelProperty(value = {"本月治理量(萬方)","整理地形綠化量"},index = 10)
    @ColumnWidth(15)
    private String afforestnum;
 
    @ExcelProperty(value = {"本月治理量(萬方)","臨時覆蓋或綠化量"},index = 11)
    @ColumnWidth(15)
    private String temporarilynum ;
 
    @ExcelProperty(value = {"本月治理量(萬方)","合計"},index = 12)
    private String goverytotal;
 
    @ExcelIgnore
    private String qynum;
 
    @ExcelIgnore
    @Dict(dicCode = "sourcestr")
    private String sourcestr;
 
    @ExcelIgnore
    private String createbyname;
 
}

3、controller

 
    
    @postMapping("pilebodystatisticsmonthexport")
    public WebApiResponse<List<PilebodycheckMonthDto>> pilebodystatisticsmonthexport (HttpServletResponse response,String month) throws IOException { 
        List<PilebodycheckMonthDto> pilebodysList = pilebodycheckService.pilebodystatisticsmonth(sysDepartDto, month);
       //設置序號
        for (int i = 1;i <= pilebodysList.size();i++){
            pilebodysList.get(i-1).setOrderNum(i+"");
        }
        response.setContentType("application/vnd.ms-excel");
        response.setCharacterEncoding("utf-8");
        // 這里URLEncoder.encode可以防止中文亂碼 當然和easyexcel沒有關系
        String fileName = URLEncoder.encode("存量建築垃圾堆體治理進度月報表", "UTF-8");
        response.setHeader("Content-disposition",  "attachment;filename=" + fileName + ".xls");
        //內容樣式策略
        WriteCellStyle contentWriteCellStyle = new WriteCellStyle();
        //垂直居中,水平居中
        contentWriteCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);
        contentWriteCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);
        contentWriteCellStyle.setBorderLeft(BorderStyle.THIN);
        contentWriteCellStyle.setBorderTop(BorderStyle.THIN);
        contentWriteCellStyle.setBorderRight(BorderStyle.THIN);
        contentWriteCellStyle.setBorderBottom(BorderStyle.THIN);
        //設置 自動換行
        contentWriteCellStyle.setWrapped(true);
        // 字體策略
        WriteFont contentWriteFont = new WriteFont();
        // 字體大小
        contentWriteFont.setFontHeightInPoints((short) 12);
        contentWriteCellStyle.setWriteFont(contentWriteFont);
        //頭策略使用默認
        WriteCellStyle headWriteCellStyle = new WriteCellStyle();
 
        //excel如需下載到本地,只需要將response.getOutputStream()換成File即可(注釋掉以上response代碼)
        EasyExcel.write(response.getOutputStream(), PilebodycheckMonthDto.class)
                //設置輸出excel版本,不設置默認為xlsx
                .excelType(ExcelTypeEnum.XLS).head(PilebodycheckMonthDto.class)
                //設置攔截器或自定義樣式
                .registerWriteHandler(new MonthSheetWriteHandler())
                .registerWriteHandler(new HorizontalCellStyleStrategy(headWriteCellStyle,contentWriteCellStyle))
                .sheet("存量建築垃圾堆體治理進度月報表")
                //設置默認樣式及寫入頭信息開始的行數
                .useDefaultStyle(true).relativeHeadRowIndex(3)
                //這里的addsumColomn方法是個添加合計的方法,可刪除
                .doWrite(pilebodycheckService.addSumColomn(pilebodysList));
        return new WebApiResponse(200, "生成excel文件成功", null);
 
    }

4、攔截器

package com.jpxx.admin.pilebody.web.api;
 
import com.alibaba.excel.write.handler.SheetWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteWorkbookHolder;
import org.apache.poi.ss.usermodel.*;
import org.apache.poi.ss.util.CellRangeAddress;
 
public class MonthSheetWriteHandler implements SheetWriteHandler {
    @Override
    public void beforeSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {
 
    }
 
    @Override
    public void afterSheetCreate(WriteWorkbookHolder writeWorkbookHolder, WriteSheetHolder writeSheetHolder) {
        Workbook workbook = writeWorkbookHolder.getWorkbook();
        Sheet sheet = workbook.getSheetAt(0);
        Row row1 = sheet.createRow(0);
        row1.setHeight((short) 500);
        Cell cell = row1.createCell(0);
        //設置單元格內容
        cell.setCellValue("附件2");
        //設置標題
        Row row2 = sheet.createRow(1);
        row2.setHeight((short) 800);
        Cell cell1 = row2.createCell(0);
        cell1.setCellValue("存量建築垃圾堆體治理進度月報表");
        CellStyle cellStyle = workbook.createCellStyle();
        cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);
        cellStyle.setAlignment(HorizontalAlignment.CENTER);
        Font font = workbook.createFont();
        font.setBold(true);
        font.setFontHeight((short) 400);
        cellStyle.setFont(font);
        cell1.setCellStyle(cellStyle);
        sheet.addMergedRegionUnsafe(new CellRangeAddress(1, 1, 0, 17));
        //設置填表日期,填報人,聯系方式
        Row row3 = sheet.createRow(2);
        row3.setHeight((short) 500);
        row3.createCell(1).setCellValue("填表日期");
        row3.createCell(11).setCellValue("填表人");
        row3.createCell(15).setCellValue("聯系方式");
 
    }
}

生成表格如下:

 


倘若要進行多sheet導出,可參考以下導出方式(僅為示例,使用時按現實要求修改參數即可) 

response.setContentType("application/vnd.ms-excel");
        response.setCharacterEncoding("utf-8");
        String fileName = URLEncoder.encode("測試輸出", "UTF-8");
        response.setHeader("Content-disposition", "attachment;filename=" + fileName + ".xls");
        //內容樣式策略
        WriteCellStyle contentWriteCellStyle = ExcelContentStyle.getWriteCellStyle();
        //頭策略使用默認
        WriteCellStyle headWriteCellStyle = new WriteCellStyle();
        try {
            ExcelWriter build = EasyExcel.write(response.getOutputStream()).build();
            WriteSheet sheet0 = EasyExcel.writerSheet(0, "一類")
                    .head(LivelihoodStatisticsYiDto.class)
                    .registerWriteHandler(new HorizontalCellStyleStrategy(headWriteCellStyle, contentWriteCellStyle))
                    .useDefaultStyle(true)
                    .registerWriteHandler(new LiveSheetWriteHandler())
                    .relativeHeadRowIndex(1)
                    .build();
            build.write(lxones, sheet0);
 
 
            WriteSheet sheet1 = EasyExcel.writerSheet(1, "二類")
                    .registerWriteHandler(new HorizontalCellStyleStrategy(headWriteCellStyle, contentWriteCellStyle))
                    .head(LivelihoodStatisticsDto.class)
                    .useDefaultStyle(true)
                    .registerWriteHandler(new LiveTowSheetWriteHandler())
                    .relativeHeadRowIndex(1).build();
            build.write(lxtwo, sheet1);
 
            build.finish();
 
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

結果圖:

 最后附上一個easyexcel工具類

import com.alibaba.excel.EasyExcel;
import com.alibaba.excel.ExcelWriter;
import com.alibaba.excel.write.builder.ExcelWriterBuilder;
import com.alibaba.excel.write.builder.ExcelWriterSheetBuilder;
import com.example.code.bot_monomer.config.common.LocalDateTimeConverter;

import org.apache.commons.lang3.StringUtils;

import java.io.IOException;
import java.io.UnsupportedEncodingException;
import java.lang.reflect.Field;
import java.net.URLEncoder;
import java.nio.charset.StandardCharsets;
import java.time.LocalDateTime;
import java.time.format.DateTimeFormatter;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

import javax.servlet.http.HttpServletResponse;

import io.swagger.annotations.ApiModel;

/**
 * Excel
 * @author shf
 * @date 2021-1-9 17:15:23
 */
public class EasyExcelUtil {
    /**
     * 導出
     * @param res {@link HttpServletResponse}
     * @param data {@link List}<{@link T}>:為空時,無法獲取表頭;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-1-9 17:18:02
     */
    @SuppressWarnings("unchecked")
    public static <T> WriterSheetBuilder write(HttpServletResponse res, List<T> data) throws IOException {
        Class<T> c = null;
        if (data != null && !data.isEmpty()) {
            T t = data.get(0);
            c = (Class<T>) t.getClass();
        }
        return write(res, data, c);
    }
    /**
     * 導出
     * @param res {@link HttpServletResponse}
     * @param data {@link List}<{@link T}>
     * @param c {@link Class}<{@link T}>:獲取配置信息的注解類;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-1-9 17:18:02
     */
    public static <T> WriterSheetBuilder write(HttpServletResponse res, List<T> data, Class<T> c) throws IOException {
        return write(res, data, c, c == null ? StringUtils.EMPTY : desc(c));
    }
    /**
     * 導出
     * @param res {@link HttpServletResponse}
     * @param data {@link List}<{@link T}>:為空時,無法獲取表頭;
     * @param name {@link String}:名稱;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-1-9 17:18:02
     */
    @SuppressWarnings("unchecked")
    public static <T> WriterSheetBuilder write(HttpServletResponse res, List<T> data, String name) throws IOException {
        Class<T> c = null;
        if (data != null && !data.isEmpty()) {
            T t = data.get(0);
            c = (Class<T>) t.getClass();
        }
        return write(res, data, c, name);
    }
    /**
     * 導出
     * @param res {@link HttpServletResponse}
     * @param data {@link List}<{@link T}>
     * @param c {@link Class}<{@link T}>:獲取配置信息的注解類;
     * @param name {@link String}:名稱;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-1-9 17:18:02
     */
    public static <T> WriterSheetBuilder write(HttpServletResponse res, List<T> data, Class<T> c, String name) throws IOException {
        return write(res, data, c, name + "-" + LocalDateTime.now().format(DateTimeFormatter.ofPattern(DateUtil.format6)), name);
    }
    /**
     * 導出
     * @param res {@link HttpServletResponse}
     * @param data {@link List}<{@link T}>
     * @param c {@link Class}<{@link T}>:獲取配置信息的注解類;
     * @param fnm {@link String}:文件名;
     * @param sheetNm {@link String}:sheet頁名稱;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-1-9 17:18:02
     */
    public static <T> WriterSheetBuilder write(HttpServletResponse res, List<T> data, Class<T> c, String fnm, String sheetNm) throws IOException {
        WriterSheetBuilder ws = create(res, c, fnm, sheetNm);
        ws.doWrite(data);
        return ws;
    }

    /**
     * 創建文件
     * @param res {@link HttpServletResponse}
     * @param c {@link Class}<{@link T}>:獲取配置信息的注解類;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-3-29 10:02:02
     */
    public static <T> WriterSheetBuilder create(HttpServletResponse res, Class<T> c) throws IOException {
        return create(res, c, c == null ? StringUtils.EMPTY : desc(c));
    }
    /**
     * 創建文件
     * @param res {@link HttpServletResponse}
     * @param c {@link Class}<{@link T}>:獲取配置信息的注解類;
     * @param name {@link String}:名稱;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-3-29 10:02:02
     */
    public static <T> WriterSheetBuilder create(HttpServletResponse res, Class<T> c, String name) throws IOException {
        return create(res, c, name + "-" + LocalDateTime.now().format(DateTimeFormatter.ofPattern(DateUtil.format6)), name);
    }
    /**
     * 創建文件
     * @param res {@link HttpServletResponse}
     * @param c {@link Class}<{@link T}>:獲取配置信息的注解類;
     * @param fnm {@link String}:文件名;
     * @param sheetNm {@link String}:sheet頁名稱;
     * @return {@link WriterSheetBuilder}
     * @author shf
     * @throws IOException
     * @date 2021-3-29 10:02:02
     */
    public static <T> WriterSheetBuilder create(HttpServletResponse res, Class<T> c, String fnm, String sheetNm) throws IOException {
        ExcelWriterBuilder ew = EasyExcel.write(resHead(res, fnm).getOutputStream(), c);
        ew.registerConverter(LocalDateTimeConverter.get());
        return new WriterSheetBuilder(ew.sheet(sheetNm));
    }

    /**
     * 響應頭
     * @param res {@link HttpServletResponse}
     * @param fnm {@link String}
     * @return {@link HttpServletResponse}
     * @throws UnsupportedEncodingException
     * @author shf
     * @date 2021-3-29 10:37:39
     */
    private static HttpServletResponse resHead(HttpServletResponse res, String fnm) throws UnsupportedEncodingException {
        if (StringUtils.isBlank(fnm)) {
            fnm = LocalDateTime.now().format(DateTimeFormatter.ofPattern(DateUtil.format6));
        }
        String utf8 = StandardCharsets.UTF_8.name();
        String encFnm = URLEncoder.encode(fnm, utf8) + ".xls";
        res.setCharacterEncoding(utf8);
        res.setContentType("application/vnd.ms-excel; charset=" + utf8);
        res.setHeader("Content-Disposition", "attachment; filename=" + encFnm + "; filename*=" + utf8 + "''" + encFnm);
        return res;
    }
    /**
     * 獲取{@link ApiModel}描述
     * @param c {@link Class}<?>
     * @return {@link String}
     * @author shf
     * @date 2021-1-26 15:52:05
     */
    private static String desc(Class<?> c) {
        if (c == Object.class) {
            return null;
        }
        ApiModel anno = c.getAnnotation(ApiModel.class);
        if (anno != null) {
            return anno.description();
        }
        return desc(c.getSuperclass());
    }

    /**
     * Sheet裝飾器
     * @author shf
     * @date 2021-3-29 16:56:53
     */
    public static class WriterSheetBuilder extends ExcelWriterSheetBuilder {
        private ExcelWriter excelWriter;

        public WriterSheetBuilder(ExcelWriterSheetBuilder ws) {
            super();
            Map<Field, Object> m = this.cp(ws);
            try {
                this.excelWriter = (ExcelWriter) m.get(ExcelWriterSheetBuilder.class.getDeclaredField("excelWriter"));
            } catch (NoSuchFieldException | SecurityException e) {
            }
        }

        public ExcelWriter getExcelWriter() {
            return this.excelWriter;
        }

        private Map<Field, Object> cp(ExcelWriterSheetBuilder ws) {
            if (ws == null) {
                return null;
            }
            Class<? extends ExcelWriterSheetBuilder> c = ws.getClass();
            Field[] fs = c.getDeclaredFields();
            Map<Field, Object> r = new HashMap<>(fs.length);
            for (int i = 0; i < fs.length; i++) {
                Field f = fs[i];
                f.setAccessible(true);
                try {
                    Object v = f.get(ws);
                    f.set(this, v);
                    r.put(f, v);
                } catch (IllegalArgumentException | IllegalAccessException e) {
                }
            }
            return r;
        }
    }
}
View Code

 

 

參考文章:

https://blog.csdn.net/qq_41514643/article/details/106993760


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM