Skip to content

Latest commit

 

History

History
323 lines (268 loc) · 10.6 KB

README.md

File metadata and controls

323 lines (268 loc) · 10.6 KB

easyexcel-util

本项目基于阿里easyexcel,在此基础上做了更进一步的封装,使其写入数据更加便捷,通过抽离出的ExcelDataHandler接口更容易处理每个cell的字体与样式

相关文档

Maven包引入

<dependency>
    <groupId>com.github.aifeinik</groupId>
    <artifactId>easyexcel-util</artifactId>
    <version>1.1</version>
</dependency>

自定义注解 ExcelValueFormat

通过该注解更加方便的处理每个数据的具体格式, 内部采用MessageFormat.format进行数据格式化

@Data
public class CampaignModel extends BaseRowModel implements Serializable {

    @ExcelProperty(value = "日期", index = 0)
    private String day;

    @ExcelProperty(value = "广告系列 ID", index = 1)
    private String campaignId;

    @ExcelProperty(value = "广告系列", index = 2)
    private String campaignName;

    @ExcelProperty(value = "费用", index = 3)
    @ExcelValueFormat(format = "{0}$")
    private String cost;

    @ExcelProperty(value = "点击次数", index = 4)
    private String clicks;

    @ExcelProperty(value = "点击率", index = 5)
    @ExcelValueFormat(format = "{0}%")
    private String ctr;

}

通过实现 ExcelDataHandler 接口来设置具体每个cell的样式与字体,如:

/**
 * Excel数据处理
 *
 * @author Feinik
 */
public interface ExcelDataHandler {

    /**
     * Excel head头部字体设置
     * @param cellIndex 列索引
     * @return CellStyle
     */
    CellStyle headFont(int cellIndex);

    /**
     * Excel 除head外的内容字体设置
     * @param cellIndex 列索引
     * @param data 行数据对象
     * @return CellStyle
     */
    CellStyle contentFont(int cellIndex, Object data);

    /**
     * Excel sheet
     * @param sheetIndex sheet索引
     * @param sheet
     */
    void sheet(int sheetIndex, Sheet sheet);

    /**
     * workbook context 初始化回调一次
     * @param workbook
     */
    void workbookContext(Workbook workbook);
}

public class CampaignDataHandler implements ExcelDataHandler {

    //通过对象池的方式来解决大数据量下重复创建CellStyle而导致的异常
    //相同设置属性的CellStyle对象达到可复用
    private Map<String, CellStyle> stylePool = new HashMap<>();

    private enum ObjType {
        HEAD_STYLE,
        INDEX4_STYLE,
        DEFAULT_STYLE
    }

    @Override
    public void workbookContext(Workbook workbook) {
        putStylePool(workbook);
    }

    //初始化对象池
    private void putStylePool(Workbook workbook) {
        stylePool.put(ObjType.HEAD_STYLE.name(), setAndGetHeadStyle(workbook));
        stylePool.put(ObjType.INDEX4_STYLE.name(), setAndGetIndex4Style(workbook));
        stylePool.put(ObjType.DEFAULT_STYLE.name(), setAndGetDefaultStyle(workbook));
    }

    @Override
    public CellStyle headFont(int cellIndex) {
        //从对象池直接获取
        return stylePool.get(ObjType.HEAD_STYLE.name());
    }

    @Override
    public CellStyle contentFont(int cellIndex, Object data) {
        CampaignModel campaign = (CampaignModel) data;

        switch (cellIndex) {
            case 4: //这里的值为Model对象中ExcelProperty注解里的index值
                if (Long.valueOf(campaign.getClicks()) > 100) { //表示将点击次数大于100的第4列也就是点击次数列的cell字体标记为红色
                    return stylePool.get(ObjType.INDEX4_STYLE.name());

                }

            default:
                return stylePool.get(ObjType.DEFAULT_STYLE.name());

        }
    }

    @Override
    public void sheet(int sheetIndex, Sheet sheet) {
        System.out.println("sheetIndex = [" + sheetIndex + "]");
    }

    public CellStyle getCellStyle(Workbook workbook) {
        CellStyle cellStyle = workbook.createCellStyle();
        cellStyle.setAlignment(HorizontalAlignment.CENTER);
        cellStyle.setVerticalAlignment(VerticalAlignment.CENTER);
        return cellStyle;
    }

    public CellStyle setAndGetHeadStyle(Workbook book) {
        final Font font = book.createFont();
        final CellStyle style = getCellStyle(book);

        font.setColor(IndexedColors.WHITE.getIndex());
        font.setBold(true);
        font.setFontHeightInPoints((short) 10);
        font.setFontName("微软雅黑");

        style.setFillPattern(FillPatternType.SOLID_FOREGROUND);
        style.setFillForegroundColor(IndexedColors.ROYAL_BLUE.getIndex());
        style.setFont(font);
        return style;
    }

    public CellStyle setAndGetDefaultStyle(Workbook book) {
        final Font font = book.createFont();
        final CellStyle style = getCellStyle(book);
        font.setFontName("微软雅黑");
        font.setFontHeightInPoints((short) 10);
        style.setFont(font);
        return style;
    }

    public CellStyle setAndGetIndex4Style(Workbook book) {
        final Font font = book.createFont();
        final CellStyle style = getCellStyle(book);
        font.setColor(IndexedColors.RED.getIndex());
        font.setFontName("微软雅黑");
        font.setFontHeightInPoints((short) 10);
        style.setFont(font);
        return style;
    }
}

Excel数据写入

小数据量一次性写入单个sheet,使用默认样式

public class ExcelTest {

    CampaignModel m1 = new CampaignModel("2019-01-01", "10000000", "campaign1", "12.21", "100", "0.11");
    CampaignModel m2 = new CampaignModel("2019-01-02", "12000010", "campaign2", "13", "99", "0.91");
    CampaignModel m3 = new CampaignModel("2019-01-03", "12001010", "campaign3", "10", "210", "1.13");
    CampaignModel m4 = new CampaignModel("2019-01-04", "15005010", "campaign4", "21.9", "150", "0.15");

    ArrayList<CampaignModel> data1 = Lists.newArrayList(m1, m2);
    ArrayList<CampaignModel> data2 = Lists.newArrayList(m3, m4);

    @Test
    public void writeExcelWithOneSheet() throws Exception {
        ExcelUtil.writeExcelWithOneSheet(new File("G:/tmp/campaign.xlsx"),
                "campaign",
                data1);
    }
}

s1

小数据量一次性写入单个sheet,使用自定义样式

    @Test
    public void writeExcelWithOneSheet2() throws Exception {
        ExcelUtil.writeExcelWithOneSheet(new File("G:/tmp/campaign.xlsx"),
                "campaign",
                data1,
                new CampaignDataHandler());
    }

s2

小数据量一次性写入多个sheet,默认样式

    @Test
    public void writeExcelWithMultiSheet() throws Exception {
        Map<String, List<? extends BaseRowModel>> map = new HashMap<>();
        map.put("sheet1", data1);
        map.put("sheet2", data2);

        ExcelUtil.writeExcelWithMultiSheet(new File("G:/tmp/campaign.xlsx"), map);
    }

s3

小数据量一次性写入多个sheet,使用自定义样式

    @Test
    public void writeExcelWithMultiSheet2() throws Exception {
        Map<String, List<? extends BaseRowModel>> map = new HashMap<>();
        map.put("sheet1", data1);
        map.put("sheet2", data2);

        ExcelUtil.writeExcelWithMultiSheet(new File("G:/tmp/campaign.xlsx"), map, new CampaignDataHandler());
    }

s3

大数据量分批写入单个sheet

    /**
     * 单个sheet
     * 测试分批写入excel文件,可通过该方式写入超大数据,而不至于一次写入大数据量导致OOM问题
     */
    @Test
    public void writeOneSheetWithWrapWriter() {
        ExcelWrapWriter wrapWriter = null;
        try {
            OutputStream os = new FileOutputStream("G:/tmp/campaign.xlsx");
            //默认样式
            //wrapWriter = new ExcelWrapWriter(os, ExcelTypeEnum.XLSX);

            //自定义excel样式
            wrapWriter = new ExcelWrapWriter(os, ExcelTypeEnum.XLSX, new CampaignDataHandler());

            List<CampaignModel> models1 = Lists.newArrayList(m1, m2);
            List<CampaignModel> models2 = Lists.newArrayList(m3, m4);

            //第一批次写入设置包含head头
            ExcelUtil.writeExcelWithOneSheet(wrapWriter, "sheet1", true, models1);

            //第二批次开始不需要在写入head头
            ExcelUtil.writeExcelWithOneSheet(wrapWriter, "sheet1", false, models2);
        } catch (Exception e) {
            e.printStackTrace();

        } finally {
            //close IO
            if (wrapWriter != null) {
                wrapWriter.finish();
            }
        }
    }

大数据量分批写入多个sheet

    /**
     * 多个sheet
     * 测试分批写入excel文件,可通过该方式写入超大数据,而不至于一次写入大数据量导致OOM问题
     */
    @Test
    public void writeMultiSheetWithWrapWriter() {
        ExcelWrapWriter wrapWriter = null;
        try {
            //os流不需要单独close,可通过wrapWriter.finish()来关闭
            OutputStream os = new FileOutputStream("G:/tmp/campaign.xlsx");
            //默认样式
            //wrapWriter = new ExcelWrapWriter(os, ExcelTypeEnum.XLSX);

            //自定义excel样式
            wrapWriter = new ExcelWrapWriter(os, ExcelTypeEnum.XLSX, new CampaignDataHandler());
            Map<String, List<? extends BaseRowModel>> batch1 = new HashMap<>();
            List<CampaignModel> models1 = Lists.newArrayList(m1, m2);
            List<CampaignModel> models2 = Lists.newArrayList(m3, m4);
            batch1.put("sheet1", models1);
            batch1.put("sheet2", models2);

            Map<String, List<? extends BaseRowModel>> batch2 = new HashMap<>();
            List<CampaignModel> models3 = Lists.newArrayList(m4, m2);
            List<CampaignModel> models4 = Lists.newArrayList(m3, m1);
            batch2.put("sheet1", models3);
            batch2.put("sheet2", models4);

            //第一批次写入设置包含head头
            ExcelUtil.writeExcelWithMultiSheet(wrapWriter, true, batch1);

            //第二批次开始不需要在写入head头
            ExcelUtil.writeExcelWithMultiSheet(wrapWriter, false, batch2);
        } catch (Exception e) {
            e.printStackTrace();

        } finally {
            //close IO
            if (wrapWriter != null) {
                wrapWriter.finish();
            }
        }
    }

测试代码

ExcelTest