首页 > 其他分享 >EasyExcel导出合并单元格

EasyExcel导出合并单元格

时间:2024-09-30 18:03:37浏览次数:6  
标签:int List EasyExcel 单元格 导出 cell add curRowIndex

处理结果:把a,b列相同内容的单元格进行合并

引入easyexcel:
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>easyexcel</artifactId>
<version>3.2.1</version>
</dependency>

示例代码:
public void exportStrategyDetail(User user, StrategyQuery query, HttpServletResponse response) throws IOException {
String sheetName = "合并单元格";
List<Map> strategyDetail = "对应的导出数据集合";

String excelName = sheetName + System.currentTimeMillis();

// 配置响应和输出流
response.setCharacterEncoding("UTF-8");
response.setHeader("Content-Type", "application/vnd.ms-excel");
response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(excelName + ExcelTypeEnum.XLSX.getValue(), "UTF-8"));

try (OutputStream out = response.getOutputStream()) {
// 将数据导出
List<List<Object>> dataList = new ArrayList<>();
for(Map dto : strategyDetail){
Object object = dto.get("children");
List<Map> childrenMaps = (List<Map>) object;
for(Map dto1 : childrenMaps){
Object object1 = dto1.get("children");
List<Map> childrenMaps1 = (List<Map>) object1;
for(Map dto2 : childrenMaps1){
List<Object> list = new ArrayList<>();
list.add(dto.get("a"));
list.add(dto1.get("b"));
list.add(dto2.get("c");
list.add(dto2.get("d"));
list.add(dto2.get("e"));
list.add(dto2.get("f"));
list.add(dto2.get("g"));
dataList.add(list);
}
}
}

ExcelWriterBuilder writerBuilder = EasyExcel.write(out);
// 添加自定义的合并单元格处理器
// 需要合并的列
int[] cols = {0, 1};
// 从第一行后开始合并
int row = 1;
writerBuilder.registerWriteHandler(new MergeCellWriteHandler(row, cols));
List<List<String>> headers = getStrategyDetailHeaders();
// 导出数据
writerBuilder.sheet(sheetName).head(headers).doWrite(dataList);
} catch (Exception e) {
// 处理异常
e.printStackTrace();
throw new IOException("导出Excel文件失败", e);
}
}
private List<List<String>> getStrategyDetailHeaders() {
List<List<String>> headers = new ArrayList<>();
headers.add(generateHeader("a"));
headers.add(generateHeader("b"));
headers.add(generateHeader("c"));
headers.add(generateHeader("d"));
headers.add(generateHeader("e"));
headers.add(generateHeader("f"));
headers.add(generateHeader("g"));
return headers;
}
private List<String> generateHeader(String... titles) {
List<String> header = new ArrayList<>();
for (String title : titles) {
header.add(title);
}
return header;
}



//自定义合并方法:MergeCellWriteHandler
import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.metadata.data.WriteCellData;
import com.alibaba.excel.write.handler.CellWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteTableHolder;
import org.apache.poi.ss.usermodel.Cell;
import org.apache.poi.ss.usermodel.CellType;
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;
public class MergeCellWriteHandler implements CellWriteHandler {

// 要合并的列索引数组
private final int[] mergeColumnIndex;
// 合并开始的行索引
private final int mergeRowIndex;

/**
* 构造函数
*
* @param mergeRowIndex 合并开始的行索引
* @param mergeColumnIndex 要合并的列索引数组
*/
public MergeCellWriteHandler(int mergeRowIndex, int[] mergeColumnIndex) {
this.mergeRowIndex = mergeRowIndex;
this.mergeColumnIndex = mergeColumnIndex;
}

@Override
public void beforeCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Row row, Head head, Integer columnIndex, Integer relativeRowIndex, Boolean isHead) {
// 单元格创建前的处理(这里不需要处理)
}

@Override
public void afterCellCreate(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {
// 单元格创建后的处理(这里不需要处理)
}

@Override
public void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {
// 当前行索引
int curRowIndex = cell.getRowIndex();
// 当前列索引
int curColIndex = cell.getColumnIndex();

// 如果当前行大于合并开始行且当前列在需要合并的列中
if (curRowIndex > mergeRowIndex && isMergeColumn(curColIndex)) {
// 进行合并操作
mergeWithPrevRow(writeSheetHolder, cell, curRowIndex, curColIndex);
}
}

/**
* 检查当前列是否在需要合并的列中
*
* @param curColIndex 当前列索引
* @return 如果是需要合并的列返回true,否则返回false
*/
private boolean isMergeColumn(int curColIndex) {
for (int columnIndex : mergeColumnIndex) {
if (curColIndex == columnIndex) {
return true;
}
}
return false;
}

/**
* 当前单元格向上合并
*
* @param writeSheetHolder 当前工作表持有者
* @param cell 当前单元格
* @param curRowIndex 当前行索引
* @param curColIndex 当前列索引
*/
private void mergeWithPrevRow(WriteSheetHolder writeSheetHolder, Cell cell, int curRowIndex, int curColIndex) {
// 获取当前单元格的数据
Object curData = getCellData(cell);
// 获取前一个单元格的数据
Cell preCell = cell.getSheet().getRow(curRowIndex - 1).getCell(curColIndex);
Object preData = getCellData(preCell);

// 判断当前单元格和前一个单元格的数据以及主键是否相同
if (curData.equals(preData) && isSamePrimaryKey(cell, curRowIndex)) {
// 获取工作表
Sheet sheet = writeSheetHolder.getSheet();
// 合并单元格
mergeCells(sheet, curRowIndex, curColIndex);
}
}

/**
* 获取单元格的数据
*
* @param cell 单元格
* @return 单元格数据
*/
private Object getCellData(Cell cell) {
return cell.getCellTypeEnum() == CellType.STRING ? cell.getStringCellValue() : cell.getNumericCellValue();
}

/**
* 判断当前单元格和前一个单元格的主键是否相同
*
* @param cell 当前单元格
* @param curRowIndex 当前行索引
* @return 如果主键相同返回true,否则返回false
*/
private boolean isSamePrimaryKey(Cell cell, int curRowIndex) {
String currentPrimaryKey = cell.getRow().getCell(0).getStringCellValue();
String previousPrimaryKey = cell.getSheet().getRow(curRowIndex - 1).getCell(0).getStringCellValue();
return currentPrimaryKey.equals(previousPrimaryKey);
}

/**
* 合并单元格
*
* @param sheet 工作表
* @param curRowIndex 当前行索引
* @param curColIndex 当前列索引
*/
private void mergeCells(Sheet sheet, int curRowIndex, int curColIndex) {
// 获取已合并的区域
List<CellRangeAddress> mergeRegions = sheet.getMergedRegions();
boolean isMerged = false;

// 检查前一个单元格是否已经被合并
for (int i = 0; i < mergeRegions.size() && !isMerged; i++) {
CellRangeAddress cellRangeAddr = mergeRegions.get(i);
if (cellRangeAddr.isInRange(curRowIndex - 1, curColIndex)) {
sheet.removeMergedRegion(i);
cellRangeAddr.setLastRow(curRowIndex);
sheet.addMergedRegion(cellRangeAddr);
isMerged = true;
}
}

// 如果前一个单元格未被合并,则新增合并区域
if (!isMerged) {
CellRangeAddress cellRangeAddress = new CellRangeAddress(curRowIndex - 1, curRowIndex, curColIndex, curColIndex);
sheet.addMergedRegion(cellRangeAddress);
}
}
}

 

标签:int,List,EasyExcel,单元格,导出,cell,add,curRowIndex
From: https://www.cnblogs.com/wj123bk/p/18442280

相关文章

  • aspose.cell 合并内容一样的单元格
    ///<summary>///数据合并///</summary>///<paramname="mySheet"></param>///<paramname="columnIndex">合并的列下标</param>///<paramname="dataIndex">从哪一行开始</param>///<paramna......
  • MATLAB中数据导入与导出的全面指南
    在MATLAB中,数据的导入与导出是数据处理工作流中的两个基本步骤。导入是将外部数据加载到MATLAB工作区的过程,而导出则是将工作区中的数据保存到外部文件中。这两个步骤对于数据分析、可视化和结果共享至关重要。本文将详细介绍如何在MATLAB中进行数据的导入和导出,包括使用各......
  • java实现导出excel添加水印或excel转pdf并添加水印
    1.通过Map对象存入excel模板中获得excel表,并添加水印importorg.apache.poi.xssf.usermodel.XSSFWorkbook;publicstaticvoidexportExcelWatermark(TemplateExportParamstemplatePath,Map<String,Object>map,StringfileName,StringmarkText,HttpServletResponseres......
  • WPS JS宏单元格双击打勾
    一行代码,rg.Value2=就可以实现在任意单元格里输入等号后面的内容。如果想要在表格中加入限制,在规定范围内的单元格中随意输入。 rg.Column==限制了等号后面,输入的列rg.Cells!=限制单元格内容。rg.MergeCells==判断是否是合并单元格完整代码functionWorkbook_Sheet......
  • JupyterLab 导出含有中文的 PDF
    一般情况下我们直接使用JupyterLab导出的PDF是无法显示中文的。这是JupyterLab导出PDF的过程:graphLRA["JupyterNotebook(.ipynb)"]-->|nbconvert|B["LaTeX(.tex)"]B-->|XeLaTeX|C["PDF(.pdf)"]可以看到在导出过程中使用了nbconvert包,这个包会先将......
  • EasyExcel导出文件基本流程以及原理分析 学习笔记(持续更新)
    EasyExcel导出文件基本流程导出文件基本流程获取数据首先获得需要导出的文件的数据内容,用一个list保存List<SysStudent>list=sysStudentService.queryList(sysStudent);定义文件名给导出的文件定义一个名字,可以添加日期或者根据输入添加其他信息,保证文件名唯一S......
  • 玩机进阶教程----MTK芯片杂牌机 小品牌机型以及其他mtk设备导出分区的另外一种方法解
            在前面多期博文中都是通过工具来导出分区制作线刷包的。今天我们以另外一种方法备份系统分区。mtk芯片较多。具体机型适合哪种方法需要自测。多种方法多条思路。遇到机型善用工具。目前一些wifi网卡 点读笔以及有些其他mtk芯片设备。通常分区都较小......
  • oracle数据泵导出导入数据
    1.创建表空间createtablespacetbs_testdatafile'/yourpath/tbs_test01.dbf'size5gautoextendonnext1gmaxsizeunlimited;2.创建用户并授权createusertestidentifiedby123456defaulttablespace tbs_test temporarytablespacetempquotaunlimitedontes......
  • 提高python读写excel单元格的效率
    目录1、批量写入一列与逐个写入一列的效率对比2、批量读取一行3、批量读取一列4、批量读取二维表格5、批量写入二维表格使用pywin32读写excel,如果逐个单元格访问,会明显感觉到执行效率很低。因为pywin32其实是python重新封装了一下com接口,交互效率本身就低。试了一......
  • 精选联盟电话采集工具 抖音精选联盟商家信息导出软件教程步骤
    使用精选联盟电话采集工具导出抖音精选联盟商家信息的详细教程介绍:抖音精选联盟电话采集工具是一款功能强大的软件,可以帮助用户快速采集抖音精选联盟商家的电话信息,并进行导出。本教程将详细介绍如何使用该工具进行商家信息的导出,并附带相应的代码示例。步骤1:安装抖音精选联盟电话......