easyExcel单元格动态合并示例

复制代码
1.先按照mergekey排序,然后相同的key分组,只保留第一个对象,其他对象属性都设置为null.

2.
复制代码
ClassPathResource resource = new ClassPathResource("template/template.xlsx");
ExcelWriter excelWriter = EasyExcel.write(outputStream).withTemplate(resource.getInputStream()).build();
WriteSheet writeSheet = EasyExcel.writerSheet().registerWriteHandler(new PartBoxMergeStrategyNew(16, 27, 11, details.size())).build();

FillConfig fillConfig = FillConfig.builder().forceNewRow(Boolean.TRUE).build();
excelWriter.fill(header, writeSheet);
excelWriter.fill(details, fillConfig, writeSheet);
excelWriter.finish();
复制代码
3.

public class PartBoxMergeStrategyNew implements CellWriteHandler {

    // 需要合并的起始列索引(从0开始)
    private final int startMergeColumns;
    // 需要合并的结束列索引(从0开始)
    private final int endMergeColumns;
    // rowIndex 从0 开始
    private final int startMergeRowIndex;
    // rowIndex 从0 开始s
    private final int lastRowIndex;

    public PartBoxMergeStrategyNew(int startMergeColumns, int endMergeColumns, int startMergeRowIndex, int rowSize) {
        this.startMergeColumns = startMergeColumns;
        this.endMergeColumns = endMergeColumns;
        this.startMergeRowIndex = startMergeRowIndex;
        this.lastRowIndex = rowSize-1+startMergeRowIndex;
    }

    @Override
    public void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {

        // 跳过表头和空值单元格
        if (isHead) return;

        Sheet sheet = writeSheetHolder.getSheet();
        int col = cell.getColumnIndex();
        int rowIndex = cell.getRowIndex();
        boolean lastFlag = rowIndex == lastRowIndex;

        if(rowIndex <= startMergeRowIndex) return;

        if (col >= startMergeColumns && col <= endMergeColumns) {
            mergeAdjacentCells(sheet, cell, rowIndex, col, lastFlag);
        }
    }
    //row是从0开始的
    private void mergeAdjacentCells(Sheet sheet, Cell cell, int currentRowIndex, int currentColIndex, boolean lastFlag) {
        int firstRow = currentRowIndex;
        int lastRow = currentRowIndex;
        // 向上查找合并起始行(跳过空值)
        // 这里是要等到最后一个空值才进行合并
        Cell curr = cell.getSheet().getRow(currentRowIndex).getCell(startMergeColumns);
        while (firstRow -1 >= startMergeRowIndex) {
            Cell prev = cell.getSheet().getRow(firstRow -1).getCell(startMergeColumns);
            if(lastFlag) {
                if(getCellValue(curr)==null&&getCellValue(prev)==null) {
                    //最后一行,当前为空,前一个为空,则继续找,说明还没有找到最前面的一行不为空的,需要继续
                    firstRow--;
                }else if(getCellValue(curr)==null&&getCellValue(prev)!=null){
                    //最后一行,当前为空,前一个不为空,说明已经找到了需要合并的最前面的一行,同时当前也是空的,所以lastRow不用-1
                    firstRow--;
                     break;
                }else if(getCellValue(curr)!=null&&getCellValue(prev)!=null){
                    //最后一行,当前不为空,前一个也不为空,相邻的两行都不是空的,无需合并
                         lastRow--;                     
                        break;
                }else{
                    //最后一行,当前不为空,前一个为空,当前不为空,则说明当前行不是一伙的,所以要-1,前一行为空,说明这一行的领头羊在前面
                    firstRow--;
                   
                }
            }else{
                if(getCellValue(curr)==null) {//不是最后一行,当前为空,则需要等到不为空的行再往上面找,不需要触发合并,
                    break;
                }else{
                    if(getCellValue(prev)==null){//不是最后一行,当前不为空,前一个为空,则继续找
                        firstRow--;
                    }else{//不是最后一行,当前不为空,前一个不为空,则不用找,当前不为空,说明跟前面不是一伙的,
                         // 前一个不为空,则说明已经找到了最前面的领头的
                        firstRow--;
                        lastRow--;
                        break;
                    }
                }
            }
        }
        // cur=17 16
        // 执行合并(如果跨越多行)
        if (lastRow > firstRow) {
            sheet.addMergedRegionUnsafe(new CellRangeAddress(
                    firstRow , lastRow , currentColIndex, currentColIndex
            ));
        }
    }

    private Object getCellValue(Cell cell) {
        switch (cell.getCellType()) {
            case STRING:
                return cell.getStringCellValue();
            case NUMERIC:
                return cell.getNumericCellValue();
            case BOOLEAN:
                return cell.getBooleanCellValue();
            default:
                return null;
        }
    }

}
相关推荐
FQNmxDG4S2 小时前
Java多线程编程:Thread与Runnable的并发控制
java·开发语言
虹科网络安全3 小时前
艾体宝干货|数据复制详解:类型、原理与适用场景
java·开发语言·数据库
axng pmje4 小时前
Java语法进阶
java·开发语言·jvm
rKWP8gKv74 小时前
Java微服务性能监控:Prometheus与Grafana集成方案
java·微服务·prometheus
老前端的功夫4 小时前
【Java从入门到入土】28:Stream API:告别for循环的新时代
java·开发语言·python
qq_435287924 小时前
第9章 夸父逐日与后羿射日:死循环与进程终止?十个太阳同时值班的并行冲突
java·开发语言·git·死循环·进程终止·并行冲突·夸父逐日
小江的记录本4 小时前
【Kafka核心】架构模型:Producer、Broker、Consumer、Consumer Group、Topic、Partition、Replica
java·数据库·分布式·后端·搜索引擎·架构·kafka
yaoxin5211234 小时前
397. Java 文件操作基础 - 创建常规文件与临时文件
java·开发语言·python
极客先躯6 小时前
高级java每日一道面试题-2025年11月24日-容器与虚拟化题[Dockerj]-runc 的作用是什么?
java·oci 的命令行工具·最小可用·无守护进程·完全标准·创建容器的核心流程·runc 核心职责思维导图
用户60648767188966 小时前
AI 抢不走的技能:用 Claude API 构建自动化工作流实战
java