EasyExcel 动态指定列合并单元格

1、网上大多是通过CellWriteHandler接口实现,CellWriteHandler按单元格读写,这种效率会降低,实现代码可以自行搜索;

2、通过实现RowWriteHandler接口,按行读取,比较当前行与前一行的数据合并单元格,下面贴上代码及效果图(根据两个字段值确定列是否需要合并)

public class PurchaseReportMergeHandler implements RowWriteHandler {
    
    //  需要合并的列索引    
    private int[] mergeColumnIndex;
    //  从那行开始合并    
    private int mergeRowIndex;

    public PurchaseReportMergeHandler() {
    }

    public PurchaseReportMergeHandler(int mergeRowIndex, int[] mergeColumnIndex) {
        this.mergeRowIndex = mergeRowIndex;
        this.mergeColumnIndex = mergeColumnIndex;
    }

    @Override
    public void afterRowDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Row row, Integer relativeRowIndex, Boolean isHead) {
        if (isHead) {
            return;
        }
        int curRowIndex = row.getRowNum();
        if (curRowIndex > mergeRowIndex) {
            for (int columnIndex : mergeColumnIndex) {
                mergeWithPrevRow(writeSheetHolder, row, curRowIndex, columnIndex);
            }
        }
    }

    private void mergeWithPrevRow(WriteSheetHolder writeSheetHolder, Row row, int curRowIndex, int columnIndex) {
        Row preRow = writeSheetHolder.getCachedSheet().getRow(curRowIndex - 1);
        //  注意这里是自己的合并规则:以项目编号及前向合同编号为合并标识
        String currUniqueMerge = row.getCell(0).getStringCellValue() + "_" + row.getCell(3).getStringCellValue();
        String preUniqueMerge = preRow.getCell(0).getStringCellValue() + "_" + preRow.getCell(3).getStringCellValue();
        boolean dataBool = StringUtils.isNotEmpty(currUniqueMerge) && StringUtils.isNotEmpty(preUniqueMerge) && currUniqueMerge.equals(preUniqueMerge);

        if (dataBool) {
            Sheet sheet = writeSheetHolder.getSheet();
            List<CellRangeAddress> mergeRegions = sheet.getMergedRegions();
            boolean isMerged = false;
            for (int i = 0; i < mergeRegions.size() && !isMerged; i++) {
                CellRangeAddress cellRangeAddr = mergeRegions.get(i);
                // 若上一个单元格已经被合并,则先移出原有的合并单元,再重新添加合并单元
                if (cellRangeAddr.isInRange(curRowIndex - 1, columnIndex)) {
                    sheet.removeMergedRegion(i);
                    cellRangeAddr.setLastRow(curRowIndex);
                    sheet.addMergedRegion(cellRangeAddr);
                    isMerged = true;
                }
            }
            // 若上一个单元格未被合并,则新增合并单元
            if (!isMerged) {
                CellRangeAddress cellRangeAddress = new CellRangeAddress(curRowIndex - 1, curRowIndex, columnIndex, columnIndex);
                sheet.addMergedRegion(cellRangeAddress);
            }
        }
    }
}
        List<PurchaseReportExportResp> result = purchaseReportService.getPurchaseReportList(req);
        // sheet中需要合并的列的索引
        final int[] mergeColumnIndex = {0, 1, 2, 3,4,5};

        EasyExcel.write(response.getOutputStream(),PurchaseReportExportResp.class)
                .autoCloseStream(false)
                .sheet("采购报表")
                .registerWriteHandler(new PurchaseReportMergeHandler(0,mergeColumnIndex))
                .doWrite(result);

效果

  • 11
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值