Java导出excel合并行功能

导出的excel需要上下行相同的数据进行行合并的功能。如图显示

这里我使用的是项目框架自带的导出模板代码,是在这套模板基础之上做的修改。

java 复制代码
// 我主要演示的就是mergeRows方法的操作,dataList是导出数据的集合。
workbook = ExcelTools.expData(workbook, templateSheet);
// 合并行操作
cseQuotaManageCheckService.mergeRows(workbook, dataList);

这是mergeRows方法。

java 复制代码
// 合并行操作
    public void mergeRows(Workbook workbook, List<?> dataList) {
        if (dataList.isEmpty()) {
            return;
        }

        Sheet sheet = workbook.getSheetAt(0);
        int startRow = 2;  // 数据从第 2 行开始
        int deptNameCol = 1;  // 从第几列开始
        int benchmarkScoreCol = 2;
        int finalScoreCol = 3;
        int daunScoreCol = 4;

        int i = 0;
        while (i < dataList.size()) {
            int mergeStart = i;
            CseQuotaManageCheckVO current = (CseQuotaManageCheckVO) dataList.get(i);

            // 查找连续相同的数据行
            while (i < dataList.size() - 1) {
                CseQuotaManageCheckVO next = (CseQuotaManageCheckVO) dataList.get(i + 1);
                if (current.getDeptName().equals(next.getDeptName()) &&
                        current.getBenchmarkScore().equals(next.getBenchmarkScore()) &&
                        current.getFinalScore().equals(next.getFinalScore()) &&
                        current.getDaunScore().equals(next.getDaunScore())) {
                    i++;
                } else {
                    break;
                }
            }

            // 如果找到多个连续相同的行,则合并它们
            if (mergeStart != i) {
                mergeCellsIfNotOverlap(sheet, startRow + mergeStart, startRow + i, deptNameCol, deptNameCol);
                mergeCellsIfNotOverlap(sheet, startRow + mergeStart, startRow + i, benchmarkScoreCol, benchmarkScoreCol);
                mergeCellsIfNotOverlap(sheet, startRow + mergeStart, startRow + i, finalScoreCol, finalScoreCol);
                mergeCellsIfNotOverlap(sheet, startRow + mergeStart, startRow + i, daunScoreCol, daunScoreCol);
            }

            // 继续下一个不相同的数据行
            i++;
        }
    }

    // 解决要合并的区域已经存在合并单元格
    private void mergeCellsIfNotOverlap(Sheet sheet, int firstRow, int lastRow, int firstCol, int lastCol) {
        CellRangeAddress newRegion = new CellRangeAddress(firstRow, lastRow, firstCol, lastCol);
        for (int i = 0; i < sheet.getNumMergedRegions(); i++) {
            CellRangeAddress existingRegion = sheet.getMergedRegion(i);
            if (isOverlapping(existingRegion, newRegion)) {
                return;  // 跳过合并,因为它与现有合并区域重叠
            }
        }
        sheet.addMergedRegion(newRegion);

        // 居中操作
        applyCellStyle(sheet, newRegion);
    }

    // 对合并后的行数据内容进行居中显示操作
    private void applyCellStyle(Sheet sheet, CellRangeAddress region) {
        Workbook workbook = sheet.getWorkbook();
        CellStyle centeredStyle = workbook.createCellStyle();
        centeredStyle.setAlignment(HorizontalAlignment.CENTER);
        centeredStyle.setVerticalAlignment(VerticalAlignment.CENTER);

        for (int row = region.getFirstRow(); row <= region.getLastRow(); row++) {
            for (int col = region.getFirstColumn(); col <= region.getLastColumn(); col++) {
                Row sheetRow = sheet.getRow(row);
                if (sheetRow == null) {
                    sheetRow = sheet.createRow(row);
                }
                Cell cell = sheetRow.getCell(col);
                if (cell == null) {
                    cell = sheetRow.createCell(col);
                }
                cell.setCellStyle(centeredStyle);
            }
        }
    }

    private boolean isOverlapping(CellRangeAddress region1, CellRangeAddress region2) {
        return !(region1.getLastRow() < region2.getFirstRow() ||
                region1.getFirstRow() > region2.getLastRow() ||
                region1.getLastColumn() < region2.getFirstColumn() ||
                region1.getFirstColumn() > region2.getLastColumn());
    }
相关推荐
小王爱吃月亮糖1 分钟前
QT开发【常用控件1】-Layouts & Spacers
开发语言·前端·c++·qt·visual studio
aworkholic14 分钟前
opencv sdk for java中提示无stiching模块接口的问题
java·c++·opencv·jni·opencv4android·stiching
字节程序员19 分钟前
四种自动化测试模型实例及优缺点详解
开发语言·javascript·ecmascript·集成测试·压力测试
程序员老冯头19 分钟前
第十六章 C++ 字符串
开发语言·c++
爱学习的白杨树19 分钟前
什么是MVCC?
java·服务器·数据库
灵槐梦35 分钟前
【速成51单片机】2.点亮LED
c语言·开发语言·经验分享·笔记·单片机·51单片机
想睡觉 . 我也想睡觉 .36 分钟前
【C++算法】1.【模板】前缀和
开发语言·c++·算法
it噩梦41 分钟前
深度分析 es multi_match 中most_fields、best_fields、cross_fields区别
java·elasticsearch
好看资源平台41 分钟前
Java Web开发基础——Web应用的请求与响应机制
java
刘Java44 分钟前
Dubbo 3.x源码(28)—Dubbo服务发布导出源码(7)应用级服务接口元数据发布
java·dubbo·dubbo源码