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();
相关推荐
静心观复7 分钟前
drawio画java的uml的类图时,class和interface的区别是什么
java·uml·draw.io
Laplaces Demon9 分钟前
Spring 源码学习(十四)—— HandlerMethodArgumentResolver
java·开发语言·学习
guygg8813 分钟前
Java 无锁方式实现高性能线程
java·开发语言
ss27314 分钟前
手写Spring第7弹:Spring IoC容器深度解析:XML配置的完整指南
java·前端·数据库
青衫码上行40 分钟前
【从0开始学习Java | 第22篇】反射
java·开发语言·学习
superlls1 小时前
(Spring)Spring Boot 中 @Valid 与全局异常处理器的联系详解
java·spring boot·后端
我星期八休息1 小时前
C++智能指针全面解析:原理、使用场景与最佳实践
java·大数据·开发语言·jvm·c++·人工智能·python
摇滚侠1 小时前
Spring Boot 3零基础教程,WEB 开发 整合 Thymeleaf 笔记36
java·spring boot·笔记
大猫会长1 小时前
docker安装php+apache
java·开发语言
野生技术架构师1 小时前
JAVA 架构师面试题含答案:JVM+spring+ 分布式 + 并发编程
java·jvm·spring