复杂表头excel导出(一对多)

项目场景:复杂表头excel导出

项目场景:业务需求,列表套列表数据,在最外层导出两个列表中的数据(一对多)


使用easyExcel导出复杂表头信息(效果图如下)

在这里插入图片描述

代码如下:
放在Util包下即可

package com.risen.app.util;

import com.alibaba.excel.metadata.CellData;
import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.write.handler.CellWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteTableHolder;
import lombok.Data;
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;

/**
 * @description: 多表头excel导出合并单元格策略
 * @author:YJ
 **/
@Data
public class MergeUtil implements CellWriteHandler {
    private int[] mergeColumnIndex;
    private int mergeRowIndex;

    public MergeUtil() {
    }

    public MergeUtil(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 afterCellDataConverted(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, CellData cellData, Cell cell, Head head, Integer integer, Boolean aBoolean) {

    }

    @Override
    public void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<CellData> list, Cell cell, Head head, Integer integer, Boolean aBoolean) {
        // 当前行
        int curRowIndex = cell.getRowIndex();
        // 当前列
        int curColIndex = cell.getColumnIndex();

        if (curRowIndex > mergeRowIndex) {
            for (int i = 0; i < mergeColumnIndex.length; i++) {
                if (curColIndex == mergeColumnIndex[i]) {
                    mergeWithPrevRow(writeSheetHolder, cell, curRowIndex, curColIndex);
                    break;
                }
            }
        }
    }

   
    private void mergeWithPrevRow(WriteSheetHolder writeSheetHolder, Cell cell, int curRowIndex, int curColIndex) {
        // 获取当前行的当前列和上一行的当前列列数据,通过上一行数据是否相同进行合并
        Object curData = cell.getCellTypeEnum() == CellType.STRING ? cell.getStringCellValue() : cell.getNumericCellValue();
        Cell preCell = cell.getSheet().getRow(curRowIndex - 1).getCell(curColIndex);
        Object preData = preCell.getCellTypeEnum() == CellType.STRING ? preCell.getStringCellValue() : preCell.getNumericCellValue();

        // 比较当前行的第一列的单元格与上一行是否相同,相同合并当前单元格与上一行
        if (curData.equals(preData)) {
            Sheet sheet = writeSheetHolder.getSheet();
            List<CellRangeAddress> mergedRegions = sheet.getMergedRegions();
            boolean isMerged = false;
            for (int i = 0; i < mergedRegions.size() && !isMerged; i++) {
                CellRangeAddress cellRangeAddr = mergedRegions.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);
            }
        }
    }
}


在接口中使用Util:


                //内容策略
                WriteCellStyle headWriteCellStyle = new WriteCellStyle();

                WriteCellStyle contentWriteCellStyle = new WriteCellStyle();
                //设置 垂直居中
                contentWriteCellStyle.setHorizontalAlignment(HorizontalAlignment.CENTER);
                contentWriteCellStyle.setVerticalAlignment(VerticalAlignment.CENTER);
                HorizontalCellStyleStrategy horizontalCellStyleStrategy = new HorizontalCellStyleStrategy(headWriteCellStyle,contentWriteCellStyle);
                
                response.setContentType("application/vnd.ms-excel;charset=utf-8");
                response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(fileName, "UTF-8") + ".xls");
                EasyExcel.write(response.getOutputStream())
                        .head(header())
                        // 合并策略:合并相同数据的行。第一个参数表示从哪一行开始进行合并,由于表头占了两行,因此从第2行开始(索引从0开始)
                        // 第二个参数是指定哪些列要进行合并
                        .registerWriteHandler(new MergeUtil(2, new int[]{0, 1, 2, 3}))
                        //设置列宽
                        .registerWriteHandler(new SimpleColumnWidthStyleStrategy(22))
                        //设置内容样式(垂直居中)
                        .registerWriteHandler(horizontalCellStyleStrategy)
                        // 注意:需要先调用registerWriteHandler()再调用sheet()方法才能使合并策略生效!!!
                        .sheet("试运行记录")
                        //数据源 dataList(List<List<Object>>)双层list集合
                        .doWrite(dataList);

下面这个是表头,在自己的接口中引用的内部方法,相同的表头需要合并就设置一样的名字即可,表数据合并策略已经在Util中做过处理



    //导出表头
    private List<List<String>> header() {
        List<List<String>> headers = new ArrayList<>();
        headers.add(Arrays.asList("试运行日期", "试运行日期"));
        headers.add(Arrays.asList("试运行基本情况", "试运行基本情况"));
        headers.add(Arrays.asList("操作人", "操作人"));
        headers.add(Arrays.asList("更新时间", "更新时间"));
        headers.add(Arrays.asList("问题记录", "问题描述"));
        headers.add(Arrays.asList("问题记录", "提出人 "));
        headers.add(Arrays.asList("问题记录", "问题状态"));
        headers.add(Arrays.asList("问题记录", "解决时间"));
        headers.add(Arrays.asList("问题记录", "解决人"));
        headers.add(Arrays.asList("问题记录", "解决方法"));
        return headers;
    }

  • 1
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,关于Java导出excel实现一对多复杂表头导出代码,以下是一个简单的示例代码: 1. 首先,需要创建一个POJO类来存储表格数据,例如ExcelData.java: ``` public class ExcelData { private String headerName; private List<String> columnNames; private List<List<String>> tableData; // 构造函数,getter和setter方法省略 } ``` 2. 为了实现一对多复杂表头,我们需要使用POI库来操作Excel文件,可以先在pom.xml文件中引入POI依赖: ``` <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>4.1.2</version> </dependency> ``` 3. 接着,可以编写导出Excel的方法: ``` public static void exportExcel(List<ExcelData> dataList, String fileName, HttpServletResponse response) { // 设置响应头信息 response.setContentType("application/octet-stream"); response.setHeader("Content-Disposition", "attachment;filename=" + fileName); // 创建Workbook对象 Workbook workbook = new XSSFWorkbook(); // 遍历数据列表 for (ExcelData data : dataList) { // 创建Sheet对象 Sheet sheet = workbook.createSheet(data.getHeaderName()); // 创建表头行 Row headerRow = sheet.createRow(0); // 创建表头单元格 Cell headerCell = headerRow.createCell(0); headerCell.setCellValue(data.getHeaderName()); sheet.addMergedRegion(new CellRangeAddress(0, 1, 0, data.getColumnNames().size() - 1)); // 创建子表头行 Row subHeaderRow = sheet.createRow(1); // 创建子表头单元格 int columnIndex = 0; for (String columnName : data.getColumnNames()) { Cell subHeaderCell = subHeaderRow.createCell(columnIndex++); subHeaderCell.setCellValue(columnName); } // 填充数据 int rowIndex = 2; for (List<String> rowData : data.getTableData()) { Row dataRow = sheet.createRow(rowIndex++); columnIndex = 0; for (String columnValue : rowData) { Cell dataCell = dataRow.createCell(columnIndex++); dataCell.setCellValue(columnValue); } } // 调整列宽 for (int i = 0; i < data.getColumnNames().size(); i++) { sheet.autoSizeColumn(i); } } // 输出Excel文件 try { OutputStream outputStream = response.getOutputStream(); workbook.write(outputStream); workbook.close(); outputStream.flush(); outputStream.close(); } catch (IOException e) { e.printStackTrace(); } } ``` 4. 最后,在Controller中调用导出Excel的方法即可: ``` @RequestMapping("/exportExcel") public void exportExcel(HttpServletResponse response) { List<ExcelData> dataList = getDataList(); // 从数据库中获取数据列表 String fileName = "test.xlsx"; // Excel文件的名称 ExcelUtil.exportExcel(dataList, fileName, response); // 导出Excel文件 } ``` 以上就是Java导出excel实现一对多复杂表头导出代码的示例。希望能对你有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值