EasyExcel简单合并单元格数据工具类

代码:

java 复制代码
package com.ly.cloud.util;

import cn.hutool.core.collection.CollUtil;
import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.write.merge.AbstractMergeStrategy;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.Sheet;
import org.apache.poi.ss.util.CellRangeAddress;

import java.util.ArrayList;
import java.util.List;
import java.util.Optional;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * @Author 
 * @Date Created in  2023/12/12 12:44
 * @DESCRIPTION: 合并单元格工具类
 * @Version V1.0
 */

public class CustomMergeStrategyUtil extends AbstractMergeStrategy {

    /**
     * 分组,每几行合并一次
     */
    public List<List<Integer>> mergeColDataGroupCountList;

    /**
     * 目标合并列index
     */
    public List<Integer> targetColumnIndex;
    /**
     * 	需要开始合并单元格的首行index
     */
    public Integer rowIndex;

    /**
     * 	mergeColDataList为待合并目标列的值
     */
    public CustomMergeStrategyUtil(List<List<String>> mergeColDataList, List<Integer> targetColumnIndex) {
        this.mergeColDataGroupCountList = getGroupCountList(mergeColDataList);
        this.targetColumnIndex = targetColumnIndex;
    }


    @Override
    public void merge(Sheet sheet, Cell cell, Head head, Integer relativeRowIndex) {

        if (null == rowIndex) {
            rowIndex = cell.getRowIndex();
        }
        // 仅从首行以及目标列的单元格开始合并,忽略其他
        if (cell.getRowIndex() == rowIndex && targetColumnIndex.contains(cell.getColumnIndex())) {
            //找到对应的需要合并的列
            AtomicInteger i = new AtomicInteger(0);
            Optional<Integer> first = targetColumnIndex.stream().filter(col -> {
                i.getAndIncrement();
                return col == cell.getColumnIndex();
            }).findFirst();
            mergeGroupColumn(sheet, first.get());
        }
    }

    public void mergeGroupColumn(Sheet sheet, Integer index) {
        int rowCount = rowIndex;
        for (Integer count : mergeColDataGroupCountList.get(index)) {
            if (count == 1) {
                rowCount += count;
                continue;
            }
            // 合并单元格
            CellRangeAddress cellRangeAddress = new CellRangeAddress(rowCount, rowCount + count - 1,
                    targetColumnIndex.get(index), targetColumnIndex.get(index));
            sheet.addMergedRegionUnsafe(cellRangeAddress);
            rowCount += count;
        }
    }

    /**
     * 	该方法将目标列根据值是否相同连续可合并,存储可合并的行数
     */
    public List<List<Integer>> getGroupCountList(List<List<String>> exportDataList) {
        if (CollUtil.isEmpty(exportDataList)) {
            return new ArrayList<>();
        }
        List<List<Integer>> groupCountListList = new ArrayList<>();
        exportDataList.forEach(dataList->{
            List<Integer> groupCountList = new ArrayList<>();
            int count = 1;
            for (int i = 1; i < dataList.size(); i++) {
                if (dataList.get(i).equals(dataList.get(i - 1))) {
                    count++;
                } else {
                    groupCountList.add(count);
                    count = 1;
                }
            }
            // 处理完最后一条后
            groupCountList.add(count);
            groupCountListList.add(groupCountList);
        });
        return groupCountListList;
    }
}

使用:

java 复制代码
        List<List<String>> mergeColDataList = Stream.of(pzry.stream().map(PzrtExportDto::getXh).collect(Collectors.toList()),
                    pzry.stream().map(PzrtExportDto::getDwmc).collect(Collectors.toList())).collect(Collectors.toList());
            //第一列(index=0)和第二列(index=1)
            List<Integer> mergeColIndexList = Stream.of(0, 1).collect(Collectors.toList());

            //内部finish的时候会自动关闭流
            EasyExcel.write(response.getOutputStream(), PzrtExportDto.class)

// 这个方法====----------》                    .registerWriteHandler(new CustomMergeStrategyUtil(mergeColDataList, mergeColIndexList))  合并单元格
                    .inMemory(Boolean.TRUE).registerWriteHandler(new CommentWriteHandler())
                    .sheet(FILE_NAME)
                    .doWrite(pzry);
相关推荐
不总是6 小时前
Windows 系统 Node.js 免安装版(zip)安装与配置教程(2026 最新)
前端·windows·node.js
不总是8 小时前
[2026最新] Windows 免安装版 MySQL 8 详细安装配置教程(ZIP 压缩包版)
数据库·windows·mysql
2401_876964139 小时前
27考研余炳森概率论|喻老李良2027资料网课
windows·git·考研·svn·eclipse·github·概率论
介一安全9 小时前
【漏洞学习】PHP+Windows环境通用文件上传漏洞深度剖析
windows·web安全·php·文件上传·安全性测试
思麟呀9 小时前
C++工业级日志项目(六)异步日志器
linux·c++·windows
fastjson_10 小时前
Win11 关闭拖动窗口自动出现的贴靠窗口分栏
windows
早睡身体真不戳10 小时前
【无标题】
java·服务器·windows
不总是11 小时前
JDK17在Windows 系统 安装与环境变量配置
windows
fastjson_12 小时前
Windows11 无法删除文件,提示:你需要 SYSTEM 提供的权限才能对此文件进行更改
windows
Face13 小时前
WSL2 初始使用
linux·windows