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();
相关推荐
毕设源码-赖学姐12 小时前
【开题答辩全过程】以 SpringMVC在筑原平面设计定制管理信息系统的应用与实践为例,包含答辩的问题和答案
java·eclipse
程序定小飞13 小时前
基于springboot的蜗牛兼职网的设计与实现
java·数据库·vue.js·spring boot·后端·spring
咖啡Beans13 小时前
RestTemplate调用API的常用写法
java·spring boot·网络协议
卷Java13 小时前
预约记录关联查询接口说明
java·开发语言·spring boot·python·微信小程序
寻星探路14 小时前
Java EE初阶启程记12---synchronized 原理
java·java-ee
qq_5746562514 小时前
java代码随想录day50|图论理论基础
java·算法·leetcode·图论
蒋星熠14 小时前
Maven项目管理与构建自动化完全指南
java·前端·python·自动化·maven
sheji341614 小时前
【开题答辩全过程】以 ssm框架的智能校园服务系统为例,包含答辩的问题和答案
java·eclipse
青云交14 小时前
Java 大视界 -- 基于 Java 的大数据实时流处理在工业物联网设备故障预测与智能运维中的应用
java·flink·kafka·工业物联网·设备故障预测·智能运维·实时流处理
fat house cat_14 小时前
为什么RocketMQ选择mmap+write?RocketMQ零拷贝技术深度解析
java·rocketmq·零拷贝