java使用EasyExcel实现合并单元格

 更新时间:2023年12月14日 08:29:59   作者:野风r  
这篇文章主要为大家详细介绍了java使用EasyExcel实现合并单元格的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下

官方默认提供了两个单元格合并策略OnceAbsoluteMergeLoopMergeStrategy
OnceAbsoluteMerge只能实现一次合并,如果需要多次合并,则需要注册多次服务。
LoopMergeStrategy可以实现循环合并,但是只能实现每隔 n 个单元格合并。

如图,还有另外一个静态类 AbstractMergeStrategy 是两者的父类,可以据此,自己实现合并方案。

合并策略

package cn.com.hmsm.bif.xingwen.accounting;  
  
import com.alibaba.excel.metadata.Head;  
import com.alibaba.excel.write.merge.AbstractMergeStrategy;  
import lombok.extern.slf4j.Slf4j;  
import org.apache.poi.ss.usermodel.Cell;  
import org.apache.poi.ss.usermodel.Row;  
import org.apache.poi.ss.usermodel.Sheet;  
import org.apache.poi.ss.util.CellRangeAddress;  
  
import java.util.List;  
  
/**  
 * 指定列,如果相邻两行的数据相同,则合并为一行  
 */  
@Slf4j  
public class ColumnSameDataMergeStrategy extends AbstractMergeStrategy {  
  
    private final int mergeRowIndex;  
  
    public ColumnSameDataMergeStrategy(int mergeRowIndex) {  
        this.mergeRowIndex = mergeRowIndex;  
    }  
  
    @Override  
    protected void merge(  
            Sheet sheet,  
            Cell cell,  
            Head head,  
            Integer relativeRowIndex  
    ) {  
  
        int rowIndex = cell.getRowIndex();  
        int colIndex = cell.getColumnIndex();  
  
        // 必须这样使用,不然lastRow 永远为空  
        sheet = cell.getSheet();  
        Row lastRow = sheet.getRow(rowIndex - 1);  
        if (lastRow == null) {  
            log.warn("rowIndex is {}, lastRow is null , data is {} ", rowIndex, cell);  
            return;  
        }  
        Cell lastRowCell = lastRow.getCell(colIndex);  
        // 先进行列的判断,再进行的判断  
        if (colIndex != mergeRowIndex) {  
            return;  
        }  
//        log.info("rowIndex is {}, data is {}", rowIndex, cell.toString());  
  
        // 如果数据相同,才进行合并  
        if (!lastRowCell.getStringCellValue().equals(cell.getStringCellValue())) {  
            return;  
        }  
  
        // 检查当前单元格是否已在合并中,如果在,则修改合数据;如果不存在,新建  
        List<CellRangeAddress> list = sheet.getMergedRegions();  
        for (int i = 0, listSize = list.size(); i < listSize; i++) {  
            CellRangeAddress cellAddresses = list.get(i);  
            // 检查列是否相同  
            if (cellAddresses.getFirstColumn() != colIndex || cellAddresses.getLastColumn() != colIndex) {  
                continue;  
            }  
            // 因为数据是从上到下添加的,行数是依次递增的,所以只需要检查合并的最后一行和上一行是否相同即可。  
            if (cellAddresses.getLastRow() == lastRowCell.getRowIndex()) {  
                // 见最后一行,改为当前行  
                cellAddresses.setLastRow(rowIndex);  
                sheet.removeMergedRegion(i);  
                sheet.addMergedRegion(cellAddresses);  
                return;  
            }  
  
        }  
  
        // 不存在已合并的区域,新建合格合并区域  
        CellRangeAddress cra = new CellRangeAddress(lastRowCell.getRowIndex(), cell.getRowIndex(), lastRowCell.getColumnIndex(), cell.getColumnIndex());  
        sheet.addMergedRegion(cra);  
  
    }  
  
}

使用

try (ExcelWriter excelWriter = EasyExcel.write(filePath).withTemplate(templateFileName).build()) {  
                    WriteSheet writeSheet = EasyExcel.writerSheet("对账单")  
                                    .registerWriteHandler(new ColumnSameDataMergeStrategy(0))  
                                    .build();  
                    Map<String, Object> map = MapUtils.newHashMap();  
                    map.put("title", title);  
                    map.put("time", reconciliationCycle);  
                    excelWriter.fill(map, writeSheet);  

                    // 这里注意 入参用了forceNewRow 代表在写入list的时候不管list下面有没有空行 都会创建一行,然后下面的数据往后移动。默认 是false,会直接使用下一行,如果没有则创建。  
                    // forceNewRow 如果设置了true,有个缺点 就是他会把所有的数据都放到内存了,所以慎用  
                    // 简单的说 如果你的模板有list,且list不是最后一行,下面还有数据需要填充 就必须设置 forceNewRow=true 但是这个就会把所有数据放到内存 会很耗内存  
                    FillConfig fillConfig = FillConfig.builder().forceNewRow(Boolean.TRUE).build();  
                    excelWriter.fill(byStoreDtoList, fillConfig, writeSheet);  
//                    excelWriter.fill(byStoreDtoList, writeSheet);  
            }

注意,在使用时,依赖的是框架提供的数据,需要设置 forceNewRowtrue,将所有数据放入内存中,一次保存。否则会出现问题。

模板

效果图

到此这篇关于java使用EasyExcel实现合并单元格的文章就介绍到这了,更多相关java EasyExcel合并单元格内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java遍历Json的简单实例

    Java遍历Json的简单实例

    这篇文章主要介绍了Java遍历Json的简单实例,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-07-07
  • 浅析Java中对称与非对称加密算法原理与使用

    浅析Java中对称与非对称加密算法原理与使用

    密码学是研究编制密码和破译密码的技术科学。这篇文章主要为大家介绍了Java中对称与非对称加密算法的原理与使用,感兴趣的小伙伴可以了解一下
    2023-03-03
  • JavaWeb实现用户登录与注册功能

    JavaWeb实现用户登录与注册功能

    这篇文章主要为大家详细介绍了JavaWeb实现用户登录与注册功能,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2021-08-08
  • Spring mvc如何实现数据处理

    Spring mvc如何实现数据处理

    这篇文章主要介绍了Spring mvc如何实现数据处理,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-03-03
  • Elasticsearch(ES)多种查询方式案例

    Elasticsearch(ES)多种查询方式案例

    Elasticsearch是一个分布式的RESTful搜索和分析引擎,可让您轻松地大规模存储,搜索和分析,这篇文章主要给大家介绍了关于Elasticsearch(ES)多种查询方式的相关资料,需要的朋友可以参考下
    2023-09-09
  • java设计模式-单例模式实现方法详解

    java设计模式-单例模式实现方法详解

    单例模式,属于创建类型的一种常用的软件设计模式。通过单例模式的方法创建的类在当前进程中只有一个实例(根据需要,也有可能一个线程中属于单例
    2021-07-07
  • Springboot如何切换默认的Tomcat容器

    Springboot如何切换默认的Tomcat容器

    这篇文章主要介绍了Springboot如何切换默认的Tomcat容器,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-06-06
  • Feign远程调用传递对象参数并返回自定义分页数据的过程解析

    Feign远程调用传递对象参数并返回自定义分页数据的过程解析

    这篇文章主要介绍了Feign远程调用传递对象参数并返回自定义分页数据的过程解析,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-03-03
  • SpringBoot实现人脸识别等多种登录方式

    SpringBoot实现人脸识别等多种登录方式

    本文主要介绍了SpringBoot实现人脸识别等多种登录方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-05-05
  • Java中空指针异常的几种解决方案

    Java中空指针异常的几种解决方案

    这篇文章主要介绍了Java中空指针异常的几种解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01

最新评论