EasyExcel 文件导出 - 合并某些列值相同的行

文章目录

EasyExcel 文件导出 - 合并某些列值相同的行

在数据处理与文件导出的过程中,我们常常会遇到各种特定的需求。今天,我们就来探讨一下使用 EasyExcel 进行文件导出时,如何合并某些列值相同的行,以实现更加高效和整洁的数据呈现。

最终效果

下面对2、3、4列进行单元格合并。

实现思路

判断当前行与上一行对应的值是否相等,若相等则进行合并操作。整个过程十分简洁明了。

创建单元格合并的策略类

注意:下面有使用lombok包,我的EasyExcel版本是4.0.2

java 复制代码
public class RowSameValueMergeStrategy extends AbstractMergeStrategy {
    /**
     * 需要合并的列位置索引
     */
    private int[] mergeColumnIndexArray;

    // 记录mergeColumnIndexArray中每一列的上一行的值和索引
    private Map<Integer, ValueAndIndex> lastValueMap;

    /**
     * 用来记录上一次的值和索引,用于判断是否需要合并
     */
    @Data
    @AllArgsConstructor
    static class ValueAndIndex {
        private String value;
        private int index;
    }

    public RowSameValueMergeStrategy(int[] mergeColumnIndexArray) {
        this.mergeColumnIndexArray = mergeColumnIndexArray;
        int initialCapacity = (int) Math.ceil(mergeColumnIndexArray.length / 0.75);
        lastValueMap = new HashMap<>(initialCapacity);
    }

    @Override
    protected void merge(Sheet sheet, Cell cell, Head head, Integer relativeRowIndex) {
        if (head == null) {
            return;
        }
        if (mergeColumnIndexArray == null) {
            return;
        }
        for (int i = 0; i < mergeColumnIndexArray.length; i++) {
            int mergeIndex = mergeColumnIndexArray[i];
            if (head.getColumnIndex() == mergeIndex) {
                String value = cell.getStringCellValue();
                int rowIndex = cell.getRowIndex();
                if (lastValueMap.get(mergeIndex) == null) {
                    lastValueMap.put(mergeIndex, new ValueAndIndex(value, rowIndex));
                }
                ValueAndIndex valueAndIndex = lastValueMap.get(mergeIndex);
                Integer lastRowIndex = valueAndIndex.getIndex();
                String lastValue = valueAndIndex.getValue();
                // 合并值相同的相邻单元格,当前单元格合并上一个单元格
                if (Objects.equals(value, lastValue) && lastRowIndex != rowIndex) {
                    sheet.addMergedRegionUnsafe(new CellRangeAddress(
                            lastRowIndex, // 起始行
                            rowIndex,     // 结束行
                            mergeIndex,   // 起始列
                            mergeIndex   // 结束列
                    ));
                }
                lastValueMap.put(mergeIndex, new ValueAndIndex(value, rowIndex));
            }
        }
    }

}

使用

使用registerWriteHandler把上面的合并单元格的策略类注册即可。

java 复制代码
         int[] mergeColumnIndexArray = {2, 3, 4}; // 要合并的列
            excelWriter = EasyExcel.write(outputStream, clazz)
                    .registerWriteHandler(new RowSameValueMergeStrategy(mergeColumnIndexArray))
                    .build();
相关推荐
澈2077 小时前
C++并查集:高效解决连通性问题
java·c++·算法
2401_873479408 小时前
运营活动被薅羊毛怎么防?用IP查询+设备指纹联动封堵漏洞
java·网络·tcp/ip·github
ShiJiuD6668889998 小时前
大事件板块一
java
摇滚侠8 小时前
@Autowired 和 @Resource 的区别
java·开发语言
SeaTunnel9 小时前
(八)收官篇 | 数据平台最后一公里:数据集成开发设计与上线治理实战
java·大数据·开发语言·白鲸开源
吴声子夜歌10 小时前
Java——线程的基本协作机制
java·线程协作
谙弆悕博士10 小时前
【附C++源码】从零开始实现 2048 游戏
java·c++·游戏·源码·项目实战·2048
独自归家的兔11 小时前
OCPP 1.6 协议详解:GetLocalListVersion 获取本地列表版本指令
java·后端·物联网·spring·ocpp1.6
Apache RocketMQ12 小时前
RocketMQ源码解析——秒级定时消息介绍
java·云原生·消息队列·rocketmq·java-rocketmq
xiaoming001812 小时前
JAVA项目打包部署运维全流程(多服务、批量)
java·linux·运维