EasyExcel - table写入复杂表头及内容

news2025/1/10 1:22:22

需求:在一个工作簿中,需要填充固定字段信息,并写入多个不同的标题列的表格及内容。
常规Excel写入一般是一个工作簿一个表头。

目录

  • 一、复杂表单分析
    • 1.表单示例
    • 2.复杂表单拆解
    • 3.准备模板
  • 二、EasyExcel文档
    • 1.最简单的填充Excel
    • 2.使用table去写入
  • 三、代码示例
    • 1.实体类
      • 1)工单信息表
      • 2)工单申请信息表
      • 3)工单设备申请明细表
      • 4)工单服务申请表
    • 2.工具类
      • 1)内容导出工具类
      • 2)合并策略
      • 3)单元格样式策略
    • 3.模板导出
  • 参考文章

一、复杂表单分析

1.表单示例

复杂表单示例1

2.复杂表单拆解

复杂表单示例拆解
示例的模板,可以拆解为6个组成部分:
(1)1-7行:表格固定的部分,需要在指定的单元格动态填充信息
(2)8-10行:表格动态写入的部分,由【工单器材】的字段列标题和内容组成。
(3)11-13行:表格动态写入的部分,由【工单服务】的字段列标题和内容组成。
(4)14行:表格动态写入的部分,由【工单结论】的字段列标题组成
(5)15-16行:表格动态写入的部分,由【费用计算】的字段列标题组成
(6)17-19行:表格固定的部分,但是因为无法往表格中间插入table,所以此处只能将固定的格式数据以table方法写入,由【表格固定结尾】的字段内容组成。

3.准备模板

因为表格的前7行是固定格式的,为了简化操作,我们直接将格式定好作为模板。而余下的行是根据数据量动态写入的。
那么,我们在代码中需要对该模板进行填充写入的操作。
在这里插入图片描述

二、EasyExcel文档

有了以上思路,开始翻阅easyExcel文档,查找可以使用的方法。最终确定使用以下两个方法去实现。

1.最简单的填充Excel

用于填充固定模板中的指定字段。
在这里插入图片描述

2.使用table去写入

用于写入动态的标题和内容。
在这里插入图片描述

三、代码示例

1.实体类

工单模板的信息来源于工单信息、工单申请信息、工单设备申请明细、工单服务申请明细表。

1)工单信息表

/**
 * 工单信息表
 */
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@Data
public class WorkOrderInfo implements Serializable {

	// 工单ID
    private Long workOrderId;

	// 工单号码
    private String workOrderCode;

    // 发起人名称
    private String creatorName;

    // 处理人名称
    private String handerName;

    // 工单标题
    private String workOrderTitle;
    
	// 工单内容
    private String workOrderContent;

   // 工单结论
    private String workOrderResult;
}

2)工单申请信息表

/**
 * 工单申请信息表
 */
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@Data
public class WorkOrderApplyInfo implements Serializable {

	// 工单申请id
    private Long workOrderApplyId;

	// 工单id
    private Long workOrderId;
    
 	// 工单上门次数
    private String visitNum;
    
	// 工单单次支付(元)
    private String singlePay;
    
 	// 工单总计费用(元)
    private String totalPay;

    // 设备申请明细
    private List<WorkOrderDevice> deviceList;

    // 服务申请明细
    private List<WorkOrderService> serviceList;
}

3)工单设备申请明细表

/**
 * 工单设备申请明细表
 */
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@Data
public class WorkOrderDevice implements Serializable {

	// 工单设备申请明细id
    private Long workOrderDeviceId;

	// 工单id
    private Long workOrderId;
    
	// 工单申请id
    private Long workOrderApplyId;
    
 	// 设备名称
    private String deviceName;
    
	// 设备明数
    private String deviceContent;
    
 	// 申请数量
    private String applyNum;
}

4)工单服务申请表

/**
 * 工单服务申请明细表
 */
@Builder
@NoArgsConstructor
@AllArgsConstructor
@ToString(callSuper = true)
@EqualsAndHashCode(callSuper = false)
@Accessors(chain = true)
@Data
public class WorkOrderService implements Serializable {

	// 工单设备申请明细id
    private Long workOrderServiceId;

	// 工单id
    private Long workOrderId;
    
	// 工单申请id
    private Long workOrderApplyId;
    
 	// 服务名称
    private String serviceName;
    
	// 服务描述
    private String serviceContent;
    
 	// 申请工时(时)
    private String hourNum;
}

2.工具类

1)内容导出工具类

因为该模板中不同的table,标题不同,占用合并的列数对应字段也不同。所以此处将设备、服务、工单结论、费用合计、固定结尾的【标题】、【字段映射列及列数】分别单独设置。最后通过【字段映射列及列数】和实际数据,构造【内容】列。

tips:
excel写入实际上是一行一行的写入数据
一行的数据结构是List<String>
多行的数据结构是List<List<String>>
/**
 * 工单确认单内容导出工具类
 */
@Component
@Slf4j
public class WorkOrderDataUtils {

	// >>>>>>>>>>>>>>> 标题构造 >>>>>>>>>>>>>>>>>>>>>>>>>>>
    /**
     * 设备列表标题
     * @return
     */
    public static List<List<String>> deviceHead() {
        List<List<String>> list = new ArrayList<List<String>>();
        List<String> head0 = new ArrayList<String>();
        head0.add("工单器材");
        // 占用3列
        list.add(head0);
        list.add(head0);
        list.add(head0);

        List<String> head1 = new ArrayList<String>();
        head1.add("器材描述");
        // 占用2列
        list.add(head1);
        list.add(head1);

        List<String> head2 = new ArrayList<String>();
        head2.add("数量");
        // 占用两列
        list.add(head2);
        list.add(head2);
        return list;
    }

	/**
     * 服务列表标题
     * @return
     */
    public static List<List<String>> serviceHead() {
        List<List<String>> list = new ArrayList<List<String>>();
        List<String> head0 = new ArrayList<String>();
        head0.add("工单服务");
        // 占用3列
        list.add(head0);
        list.add(head0);
        list.add(head0);

        List<String> head1 = new ArrayList<String>();
        head1.add("服务描述");
        // 占用2列
        list.add(head1);
        list.add(head1);

        List<String> head2 = new ArrayList<String>();
        head2.add("工时");
        // 占用两列
        list.add(head2);
        list.add(head2);
        return list;
    }

	/**
     * 工单结论标题:
     * 将结论传入作为标题
     * @return
     */
    public static List<List<String>> workOrderResultHead(String workOrderResult) {
        List<List<String>> list = new ArrayList<List<String>>();
        List<String> head0 = new ArrayList<String>();
        head0.add("工单结论");
        // 占用3列
        list.add(head0);
        list.add(head0);
        list.add(head0);

        List<String> head1 = new ArrayList<String>();
        head1.add(workOrderResult);
        // 占用4列
        list.add(head1);
        list.add(head1);
        list.add(head1);
        list.add(head1);

        return list;
    }
	/**
     * 费用合计标题
     * 将上门次数,每次支付(元),总费用传入作为标题
     * @return
     */
    public static List<List<String>> totalHead(String visitNum, String singlePay, String totalPay) {
        List<List<String>> list = new ArrayList<List<String>>();
        List<String> head0 = new ArrayList<String>();
        head0.add("上门次数");
        head0.add(visitNum);

        // 占用3列
        list.add(head0);
        list.add(head0);
        list.add(head0);

        List<String> head1 = new ArrayList<String>();
        head1.add("每次支付(元)");
        head1.add(singlePay);
        // 占用2列
        list.add(head1);
        list.add(head1);

        List<String> head2 = new ArrayList<String>();
        head2.add("总费用");
        head2.add(totalPay);
        // 占用两列
        list.add(head2);
        list.add(head2);
        return list;
    }
    
 // >>>>>>>>>>>>>>> 结尾固定内容 >>>>>>>>>>>>>>>>>>>>>>>>>>>
    /**
     * 结尾固定内容
     * @return
     */
    public static List<List<Object>> getFinalData(String content) {
        List<List<Object>> list = new ArrayList<List<Object>>();
        List<Object> row0 = new ArrayList<Object>();
        row0.add(content);
        row0.add(content);
        row0.add(content);
        row0.add(content);
        row0.add(content);
        row0.add(content);
        row0.add(content);

        List<Object> row5 = new ArrayList<Object>();
        row5.add("发起人签字");
        row5.add("");
        row5.add("");
        row5.add("");
        row5.add("处理人签字");
        row5.add("");
        row5.add("");

        list.add(row0);
        list.add(row0);
        list.add(row5);
        return list;
    }
    
 // >>>>>>>>>>>>>> 内容构造 >>>>>>>>>>>>>>>>>>>>>>>>>>>>
 /**
     * 设备字段映射及占用列数
     * @return
     */
    public static LinkedHashMap<String, Integer> fieldRowNumMapByDevice() {
        LinkedHashMap<String, Integer> fieldRowNumMap = new LinkedHashMap<>();
        // 器材名称
        fieldRowNumMap.put("deviceName", 3);
        // 器材描述
        fieldRowNumMap.put("deviceContent", 2);
        // 申请数量
        fieldRowNumMap.put("applyNum", 2);
        return fieldRowNumMap;
    }
    
 /**
     * 服务字段映射及占用列数
     * @return
     */
    public static LinkedHashMap<String, Integer> fieldRowNumMapByDevice() {
        LinkedHashMap<String, Integer> fieldRowNumMap = new LinkedHashMap<>();
        // 服务名称
        fieldRowNumMap.put("serviceName", 3);
        // 服务描述
        fieldRowNumMap.put("serviceContent", 2);
        // 服务工时
        fieldRowNumMap.put("hourName", 2);
        return fieldRowNumMap;
    }
    
// >>>>>>>>>>>>>> 内容构造 >>>>>>>>>>>>>>>>>>>>>>>>>>>>

    /**
     * 设备/服务列表内容构造
     * 通过反射获取对应字段的内容值
     * @param detailsList 设备/服务列表
     * @param fieldRowNumMap 对应的字段及列数对象
     * @return
     */
    public static List<List<Object>> contentRowData(List<?> applyList, Class<?> clazz
            , LinkedHashMap<String, Integer> fieldRowNumMap) throws NoSuchFieldException, IllegalAccessException {

        List<List<Object>> list = new ArrayList<List<Object>>();

        for (BusinessInternalDetailsExcelVO deviceDTO : detailsList) {

            List<Object> rown = new ArrayList<Object>();


            // 防止物料名称和规格列内容相同合并
            String previouFieldContent = "";

            // {"key":"productName", value:"3"} 字段和对应设置的个数,后续设置了相同内容合并的策略
            for (Map.Entry <String, Integer>  entry : fieldRowNumMap.entrySet()) {
                String fieldName = entry.getKey();
                Integer fieldRowNum = entry.getValue();

                // 通过反射,根据字段名获取该对象中的字段值
                Field declaredField = clazz.class.getDeclaredField(fieldName);
                declaredField.setAccessible(true);
                String fieldContent = " ";
                if (declaredField.get(deviceDTO) != null) {
                    // 当为空字符串时,不赋值
                    if (!"".equals(declaredField.get(deviceDTO))) {
                        fieldContent = declaredField.get(deviceDTO).toString();
                    }

                    // 防止相邻两列的内容相同合并
                    if (fieldContent.equals(previouFieldContent)) {
                        fieldContent = " " + fieldContent;
                    }
                }
                previouFieldContent = fieldContent;

                for (int i=0; i < fieldRowNum; i++) {
                    rown.add(fieldContent);
                }
            }

            list.add(rown);
        }
        return list;
    }

// >>>>>>>>>>>>>> 生成模板 >>>>>>>>>>>>>>>>>>>>>>>>>>>>

    /**
     * 工单确认单-生成模板
     */
    public static void generateWorkOrderTemplate(ExcelWriter excelWriter, WriteSheet writeSheet
            , WorkOrderApplyInfo orderApply) throws IOException {
        
        try {

            // table集成sheet配置(有/无头),此处多table需要各自的表头
            // 设备列表
            WriteTable writeTable0 = EasyExcel.writerTable(0).needHead(Boolean.TRUE).build();
            // 服务列表
            WriteTable writeTable1 = EasyExcel.writerTable(1).needHead(Boolean.TRUE).build();
            // 工单结论列
            WriteTable writeTable2 = EasyExcel.writerTable(2).needHead(Boolean.TRUE).build();
            // 费用合计
            WriteTable writeTableTotal = EasyExcel.writerTable(3).needHead(Boolean.TRUE).build();
            // 固定内容部分(备注)
            WriteTable writeTableFin = EasyExcel.writerTable(4).needHead(Boolean.FALSE).build();

            // 设备table写入头和内容
            writeTable0.setHead(this.deviceHead());
            excelWriter.write(contentRowData(orderApply.getDeviceList(), WorkOrderDevice.class, this.fieldRowNumMapByDevice())
                    , writeSheet, writeTable0);

            // 服务 table写入头和内容
            writeTable1.setHead(this.serviceHead());
            excelWriter.write(contentRowData(this.getServiceList(), WorkOrderService.class, this.fieldRowNumMapByService())
                    , writeSheet, writeTable1);

            // 工单结论 table仅写入头
            writeTable2.setHead(this.workOrderResultHead( orderApply.getWorkOrderResult()));
            excelWriter.write(new ArrayList<>(), writeSheet, writeTable2);


            // 合计 table仅写入头
            writeTableTotal.setHead(this.workOrderResultHead(orderApply.getVisitNum(), orderApply.getWorkOrderResult(), orderApply.getSinglePay(), orderApply.getTotalPay()));
            excelWriter.write(new ArrayList<>(), writeSheet, writeTableTotal);

            // 固定内容备注+签名行不创建头,在合计table之后写入数据
            excelWriter.write(this.getFinalData("备注 ....."), writeSheet, writeTableFin);

            //完成
            excelWriter.finish();
        } catch (Exception e) {
            log.error("工单确认单模板生成,方法异常>>>>>>>>>>>>>>", e);
        }
    }
}

2)合并策略

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.*;

/**
 * 功能描述:规则: 优先合并列,再合并行
 *
 * @author SXT
 * @version 1.0
 * @date 2024/3/9 15:12
 */
public class MergeCellStrategyHandler extends AbstractMergeStrategy {
    /**
     * 相同列合并
     */
    private boolean alikeColumn;

    /**
     * 相同行合并
     */
    private boolean alikeRow;

    /** 开始进行合并的行index */
    private int rowIndex;

    /** 开始进行合并的行index */
    private int rowIndexStart;

    /** 跨行合并的列index */
    private Set<Integer> columns;

    private int currentRowIndex = 0;

    /**
     * 构造方法,指定合并方式
     * @param alikeColumn
     * @param alikeRow
     * @param rowIndex
     * @param columns
     */
    public MergeCellStrategyHandler(boolean alikeColumn, boolean alikeRow, int rowIndex, Set<Integer> columns){
        this.alikeColumn = alikeColumn;
        this.alikeRow = alikeRow;
        this.rowIndex = rowIndex;
        this.columns = columns;
    }

    /**
     * 构造方法,指定合并方式
     * @param alikeColumn
     * @param alikeRow
     * @param rowIndex
     * @param columns
     */
    public MergeCellStrategyHandler(boolean alikeColumn, boolean alikeRow, int rowIndex, Set<Integer> columns, int rowIndexStart){
        this.alikeColumn = alikeColumn;
        this.alikeRow = alikeRow;
        this.rowIndex = rowIndex;
        this.columns = columns;
        this.rowIndexStart = rowIndexStart;
    }

    /**
     * 指定是否进行跨列合并单元格
     * @param alikeColumn
     * @param rowIndex
     */
    public MergeCellStrategyHandler(boolean alikeColumn, int rowIndex){
        this(alikeColumn, false, rowIndex, new HashSet<>());
    }

    /**
     * 指定是否进行跨列合并单元格
     * @param alikeColumn
     * @param rowIndex
     * @param rowIndexStart 开始行合并的行数
     */
    public MergeCellStrategyHandler(boolean alikeColumn, int rowIndex, int rowIndexStart){
        this(alikeColumn, false, rowIndex, new HashSet<>(), rowIndexStart);
    }

    /**
     * 指定是否进行跨行合并单元格
     * @param alikeRow
     * @param rowIndex
     * @param columns
     */
    public MergeCellStrategyHandler(boolean alikeRow, int rowIndex, Set<Integer> columns){
        this(false, alikeRow, rowIndex, columns);
    }


    @Override
    protected void merge(Sheet sheet, Cell cell, Head head, Integer integer) {
        int rowId = cell.getRowIndex();
        currentRowIndex = rowId == currentRowIndex ? currentRowIndex : rowId;
        if (rowIndex > rowId) {
            return;
        }

        int columnId = cell.getColumnIndex();
        // 列合并
        if (alikeColumn && columnId > 0) {
            String currentCellVal = this.getCellVal(cell);
            Cell preCell = cell.getRow().getCell(columnId - 1);
            String preCellVal = this.getCellVal(preCell);
            if (null != currentCellVal && null != preCellVal && !preCellVal.isEmpty() && !currentCellVal.isEmpty()) {
                // 当前单元格内容与上一个单元格内容相等,进行合并处理
                if (preCellVal.equals(currentCellVal)) {
                    CellRangeAddress rangeAddress = new CellRangeAddress(currentRowIndex, currentRowIndex, columnId - 1, columnId);
                    rangeAddress = this.findExistAddress(sheet, rangeAddress, currentCellVal);
                    if (null != rangeAddress) {
                        sheet.addMergedRegion(rangeAddress);
                    }
                }
            }
        }

        // 限制开始行合并的行数
        if (rowId > rowIndexStart) {
            // 行合并
            if (alikeRow && rowIndex < rowId && columns.contains(columnId)) {
                String currentCellVal = this.getCellVal(cell);
                Cell preCell = sheet.getRow(rowId - 1).getCell(columnId);
                String preCellVal = this.getCellVal(preCell);
                if (null != currentCellVal && null != preCellVal && !preCellVal.isEmpty() && !currentCellVal.isEmpty()) {
                    // 当前单元格内容与上一行单元格内容相等,进行合并处理
                    if (preCellVal.equals(currentCellVal)) {
                        //sheet.validateMergedRegions();
                        CellRangeAddress rangeAddress = new CellRangeAddress(currentRowIndex - 1, currentRowIndex, columnId, columnId);
                        rangeAddress = this.findExistAddress(sheet, rangeAddress, currentCellVal);
                        if (null != rangeAddress) {
                            sheet.addMergedRegion(rangeAddress);
                        }
                    }
                }
            }
        }


    }

    /**
     * 合并单元格地址范围,发现存在相同的地址则进行扩容合并
     *
     * @param sheet
     * @param rangeAddress  单元格合并地址
     * @param currentVal 当前单元格中的值
     * @return
     */
    private CellRangeAddress findExistAddress(Sheet sheet, CellRangeAddress rangeAddress, String currentVal) {
        List<CellRangeAddress> mergedRegions = sheet.getMergedRegions();
        int existIndex = 0;
        Map<Integer, CellRangeAddress> existIdexMap = new LinkedHashMap<>();
        if (null != mergedRegions && !mergedRegions.isEmpty()) {
            //验证当前合并的单元格是否存在重复
            for (CellRangeAddress mergedRegion : mergedRegions) {
                if (mergedRegion.intersects(rangeAddress)) {
                    existIdexMap.put(existIndex, mergedRegion);
                }
                existIndex++;
            }
        }
        if (existIdexMap.isEmpty()) {
            return rangeAddress;
        }
        List<Integer> existIndexList = new ArrayList<>(existIdexMap.size());
        for (Map.Entry<Integer, CellRangeAddress> addressEntry : existIdexMap.entrySet()) {
            CellRangeAddress exist = addressEntry.getValue();
            // 自动进行单元格合并处理
            int firstRow = rangeAddress.getFirstRow();
            int lastRow = rangeAddress.getLastRow();
            int firstColumn = rangeAddress.getFirstColumn();
            int lastColumn = rangeAddress.getLastColumn();

            int firstRow1 = exist.getFirstRow();
            int lastRow1 = exist.getLastRow();
            int firstColumn1 = exist.getFirstColumn();
            int lastColumn1 = exist.getLastColumn();
            // 跨行合并 最后一列相等, 行不相等
            if (lastRow > lastRow1 && lastColumn == lastColumn1) {
                // 检查进行跨行合并的单元格是否已经存在跨列合并
                if (lastColumn > 0 && firstColumn1 != lastColumn1) {
                    // 获取当前单元格的前一列单元格
                    String cellVal = this.getCellVal(sheet.getRow(lastRow).getCell(lastColumn - 1));
                    if (null != cellVal && cellVal.equals(currentVal)) {
                        exist.setLastRow(lastRow);
                    }
                } else {
                    exist.setLastRow(lastRow);
                }
                rangeAddress = exist;
                existIndexList.add(addressEntry.getKey());
            }

            // 跨列合并 行相等,列不相等
            if (lastColumn > lastColumn1 && firstRow == firstRow1 ) {
                exist.setLastColumn(lastColumn);
                rangeAddress = exist;
                existIndexList.add(addressEntry.getKey());
            }
        }
        // 移除已经存在且冲突的合并数据
        if (existIndexList.isEmpty()) {
            rangeAddress = null;
        }else {
            sheet.removeMergedRegions(existIdexMap.keySet());
        }
        return rangeAddress;
    }

    /**
     * 获取单元格中的内容
     * @param cell
     * @return
     */
    private String getCellVal(Cell cell) {
        String val = null;
        try {
            val = cell.getStringCellValue();
        }catch (Exception e){
            System.out.printf("读取单元格内容失败:行%d 列%d %n", (cell.getRowIndex() + 1), (cell.getColumnIndex() + 1));
        }
        return val;
    }
}

3)单元格样式策略

import com.alibaba.excel.metadata.Head;
import com.alibaba.excel.metadata.data.WriteCellData;
import com.alibaba.excel.util.StyleUtil;
import com.alibaba.excel.write.handler.CellWriteHandler;
import com.alibaba.excel.write.metadata.holder.WriteSheetHolder;
import com.alibaba.excel.write.metadata.holder.WriteTableHolder;
import com.alibaba.excel.write.metadata.style.WriteCellStyle;
import com.alibaba.excel.write.metadata.style.WriteFont;
import org.apache.poi.ss.usermodel.*;
import java.util.HashMap;
import java.util.List;

/**
 * @Desc 拦截处理单元格创建
 */
public class CellStyleWriteHandler implements CellWriteHandler {
    /**
     * map
     * key:第i行
     * value:第i行中单元格索引集合
     */
    private HashMap<Integer,List<Integer>> map;

    /**
     * 颜色
     */
    private Short colorIndex;

    /**
     * 有参构造
     * map:用来记录需要为第key行中的第value.get(i)列设置样式
     * colorIndex:表示单元格需要设置的颜色
     */
    public CellStyleWriteHandler(HashMap<Integer, List<Integer>> map, Short colorIndex) {
        this.map = map;
        this.colorIndex = colorIndex;
    }

    /**
     * 无参构造
     */
    public CellStyleWriteHandler() {

    }

    /**
     * 在创建单元格之前调用
     */
    @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) {
    }

    /**
     * 在单元上的所有操作完成后调用
     * 指定单元格特殊处理 todo 待修改为指定样式
     */
    @Override
    public void afterCellDispose(WriteSheetHolder writeSheetHolder, WriteTableHolder writeTableHolder, List<WriteCellData<?>> cellDataList, Cell cell, Head head, Integer relativeRowIndex, Boolean isHead) {

        /**
         * 考虑到导出数据量过大的情况,不对每一行的每一个单元格进行样式设置,只设置必要行中的某个单元格的样式
         */
        //当前行的第i列
        int i = cell.getColumnIndex();
        //不处理第一行
        if (0 != cell.getRowIndex()) {
            List<Integer> integers = map.get(cell.getRowIndex());
            if (integers != null && integers.size() > 0) {
                if (integers.contains(i)) {
                    // 根据单元格获取workbook
                    Workbook workbook = cell.getSheet().getWorkbook();
                    //设置行高
                    writeSheetHolder.getSheet().getRow(cell.getRowIndex()).setHeight((short) (4.5 * 256));
                    // 单元格策略
                    WriteCellStyle contentWriteCellStyle = new WriteCellStyle();
                    // 设置背景颜色白色
                    contentWriteCellStyle.setFillForegroundColor(IndexedColors.WHITE.getIndex());
                    // 设置垂直居中为向上对齐
                    contentWriteCellStyle.setVerticalAlignment(VerticalAlignment.TOP);
                    // 设置左右对齐为靠左对齐
                    contentWriteCellStyle.setHorizontalAlignment(HorizontalAlignment.LEFT);
                    // 设置单元格上下左右边框为细边框
                    contentWriteCellStyle.setBorderBottom(BorderStyle.THIN);
                    contentWriteCellStyle.setBorderLeft(BorderStyle.THIN);
                    contentWriteCellStyle.setBorderRight(BorderStyle.THIN);
                    contentWriteCellStyle.setBorderTop(BorderStyle.THIN);
                    // 创建字体实例
                    WriteFont cellWriteFont = new WriteFont();
                    // 设置字体大小
                    cellWriteFont.setFontName("宋体");
                    cellWriteFont.setFontHeightInPoints((short) 11);
                    //设置字体颜色:无效
                    cellWriteFont.setColor(colorIndex);
                    //单元格颜色:无效
                    contentWriteCellStyle.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.getIndex());
                    contentWriteCellStyle.setWriteFont(cellWriteFont);

                    CellStyle cellStyle = StyleUtil.buildCellStyle(workbook, null, contentWriteCellStyle);
                    //设置当前行第i列的样式
                    cell.getRow().getCell(i).setCellStyle(cellStyle);
                }
            }
        }
    }
}

3.模板导出

/**
 * 导出工单表单模板
 */
public void exportWorkOrderTemplate(Long orderId) throws IOException {
        /**:
         * 根据表单id查询表单信息,回填模板表
         */
        WorkOrderInfo workOrderInfo = workOrderInfoMapper.selectById(orderId);
        if (workOrderInfo == null) {
            throw new BizException("工单信息不存在");
        }

        /**
         * 查询工单设备、服务申请情况
         */
        WorkOrderApplyInfo orderApply = workOrderApplyService.getWorkOrderApplayInfo(olderId);

		// 开始合并的行数
        int addRowNum = 3;
      
        // 设备列表
        List<WorkOrderDevice> deviceList = orderApply.getDeviceList();
        // 服务列表
        List<WorkOrderService> serviceList = orderApply.getServiceList();

        // 尾部固定行开始位置 = 前置固定行(7) + 设备列表行 + 服务列表行 + 合计行
        int endCol = MictConstants.EIGHT + deviceList.size() + serviceList.size() + addRowNum;

        // >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>> 文件生成准备 >>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>>
        RequestAttributes requestAttributes = RequestContextHolder.currentRequestAttributes();
        HttpServletResponse response = ((ServletRequestAttributes) requestAttributes).getResponse();
        // 获取模版文件
        ClassPathResource classPathResource = new ClassPathResource("xls_template/work_order_template.xlsx");
        InputStream fis = classPathResource.getInputStream();

        // 文件输出流
        String fileName = URLEncoder.encode(String.format("%s.xlsx", "工单确认单"), "UTF-8");
        response.setContentType("application/x-download;charset=UTF-8");
        response.addHeader("Cache-Control", "no-cache, no-store, must-revalidate");
        response.addHeader("Pragma", "no-cache");
        response.addHeader("Access-Control-Expose-Headers", "Content-Disposition");
        response.setHeader("Content-Disposition", "attachment;filename=" + fileName);
        OutputStream out = response.getOutputStream();
        try {

            // 构造固定填充模板对象
            WorkOrderInfo workOrderPrefix = WorkOrderInfo.builder()
                    .workOrderCode(workOrderInfo.getWorkOrderCode())
                    .creatorName(workOrderInfo.getCreatorName())
                    .handerName(workOrderInfo.getHanderName())
                    .workOrderTitle(workOrderInfo.getWorkOrderTitle())
                    .workOrderContent(businessInfo.getWorkOrderContent())
.build();

            /**
             * 用来记录需要为第`key`行中的第`value.get(i)`列设置样式
             */
            // 指定需要跨行合并的列项:7列
            HashSet<Integer> colSet = new HashSet<>(Arrays.asList(0, 1, 2, 3, 4, 5, 6));
            HashMap<Integer, List<Integer>> map = new HashMap<>(MictConstants.SIX);
            for (int i = endCol; i < endCol + MictConstants.SIX; i++) {
                map.put(i, Arrays.asList(0,1,2,3,4,5,6));
            }

            // 设置excel输出策略
            ExcelWriter excelWriter = EasyExcel
                    .write(out, WorkOrderInfo.class)
                    .withTemplate(fis)
                    // 默认样式策略
                    .registerWriteHandler(EasyExcelUtils.defaultStyles())
                    // 指定单元格样式(备注说明内容栏)
                    .registerWriteHandler(new CellStyleWriteHandler(map, IndexedColors.RED.getIndex()))
                    // 行和列合并策略
                    .registerWriteHandler(new MergeCellStrategyHandler(true, true, 7, colSet, endCol))
                    .build();


            // sheet设置不需要头
            WriteSheet writeSheet = EasyExcel.writerSheet(0,"工单确认单").needHead(Boolean.FALSE).build();

            // 填充固定值
            excelWriter.fill(workOrderPrefix, writeSheet);
			// 使用table写入模板
           	WorkOrderDataUtils.generateWorkOrderTemplate(excelWriter, writeSheet, orderApply);

            out.flush();
            out.close();
            fis.close();

        } catch (Exception e) {
            log.error("工单确认单生成,方法异常>>>>>>>>>>>>>>", e);
        }
    }

参考文章

EasyExcel动态单元格合并(跨行或跨列) by 酸菜鱼没有鱼

easyExcel实现单sheet多子表,并结合动态表头,复杂表头 by IM@taoyalong

easyExcel官方文档

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1960143.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

[Linux安全运维] Nginx安装部署以及LNMP框架搭建保姆级教程

前言 LNMP&#xff1a;Linux 系统下 NginxMySQLPHP 网站服务器架构。因为四种软件均是免费开源网站&#xff0c;所有这是一个免费、高效的网站服务系统。 本章主要介绍的是Nginx相关的环境部署&#xff0c;以及LNMP框架的搭建&#xff0c;Nginx知识点介绍在文章&#xff1a;[…

昇思25天学习打卡营第23天|基于MindSpore的红酒分类实验案例:从数据准备到模型预测

目录 MindSpore 版本配置与红酒数据集下载 葡萄酒数据读取、处理与可视化分析 基于 KNN 算法的样本分类模型构建与预测函数定义 基于 KNN 模型的测试集预测与准确率计算 MindSpore 版本配置与红酒数据集下载 首先使用 %%capture captured_output 捕获后续代码的输出。然后&a…

【Django5】内置Admin系统

系列文章目录 第一章 Django使用的基础知识 第二章 setting.py文件的配置 第三章 路由的定义与使用 第四章 视图的定义与使用 第五章 二进制文件下载响应 第六章 Http请求&HttpRequest请求类 第七章 会话管理&#xff08;Cookies&Session&#xff09; 第八章 文件上传…

python中,jsonpath提取数据的时候出现TypeError: ‘bool‘ object is not subscriptable怎么解决

json格式如下&#xff1a; { success: True, result: { codeInfo: { code: 0, msg: 成功 }, uploadToken: { resId: rzJRpo, endpoint: https://sit-api-ypsx-resource.ypsx-internal.com/r…

知识分享|temu跨境选品师盈利一般要多久?

在成为一名跨境选品师&#xff0c;特别是在TEMU(The Easy Market University)平台上&#xff0c;盈利的速度取决于多种因素&#xff0c;包括个人技能、市场选择、产品定位和运营策略等。这些因素共同决定了一个选品师从初始阶段到稳定盈利的时间轨迹。 首先&#xff0c;对于新手…

vue3组件通信(一)

组件通信 一.props(父<>子)二.自定义事件&#xff08;子>父&#xff09;三.mitt(实现任意组件通信)四.v-model(父<>子)(1).v-model的本质(2).组件标签中v-model的本质(3).$event到底是什么 概况 一.props(父<>子) 使用频率最高 若 父传子&#xff1a;属性…

Redis结合Lua脚本的简单使用

我们就拿购物车举例子 现在有5个东西免费送&#xff0c;我们只能选择1个 例如 可乐 美年达 香蕉 苹果 薯片 我们选择后就放进redis里面 然后我们不能选重复&#xff0c;只能选不同 Lua脚本 我们redis使用lua脚本的时候&#xff0c;会传两个参数进去 一个是List<Strin…

(新)VMware虚拟机安装Linux教程(超详细)

创作不易&#xff0c;禁止转载抄袭&#xff01;&#xff01;&#xff01;违者必究&#xff01;&#xff01;&#xff01; 创作不易&#xff0c;禁止转载抄袭&#xff01;&#xff01;&#xff01;违者必究&#xff01;&#xff01;&#xff01; 创作不易&#xff0c;禁止转载抄…

PMP相关考点整理

PMP知识点整理 一、预测型4. 项目整合管理启动:4.1 制定项目章程规划:4.2 制定项目管理计划执行:4.3 指导与管理项目工作4.4 管理项目知识监控4.5监控项目工作4.6 实施整体变更控制【都是重点】收尾4.7 结束项目或阶段5. 项目范围管理启动:/规划:/5.1 规划范围管理5.2 收集…

36.【C语言】函数栈帧的创建和销毁

本文将解决以下问题 局部变量是怎么创建的&#xff1f; 为什么局部变量的值是随机值&#xff1f; 函数是怎么传参的&#xff1f;传参的顺序是怎样的&#xff1f; 形参和实参是什么关系&#xff1f; 函数调用是怎么做的&#xff1f; 函数调用是结束后怎么返回的&#xff1f; 本文…

电脑部分数据丢失?别担心,这里有恢复指南!

在数字化时代&#xff0c;电脑已成为我们日常生活中不可或缺的设备&#xff0c;存储着大量重要的工作文档、个人照片、视频回忆以及各类学习资料。然而&#xff0c;无论是由于误操作、系统故障、病毒攻击还是硬件损坏&#xff0c;电脑数据丢失的情况时有发生&#xff0c;令人焦…

JAVA(IO流-字节流)day 7.29

ok家人们今天继续学习IO流&#xff0c; 一.字节流 存储时&#xff0c;都是使用二进制来保存。 2.1 字节输出流OutputStream OutputStream是字节输出流的超类&#xff08;父类&#xff09;, 方法 public abstract void write(int b)&#xff1a; 一次写一个字节数据。pub…

如何实现无公网IP远程访问本地内网部署的Proxmox VE虚拟机平台

文章目录 1. 局域网访问PVE2. 安装Cpolar 工具3. 创建PVE公网地址4. 远程访问PVE5. 设置固定域名6. 固定地址访问 Proxmox VE是一个完全开源的平台&#xff0c;基于Debian Linux&#xff0c;用于运行虚拟机和容器。为了提供最大的灵活性&#xff0c;它支持两种虚拟化技术&#…

超简单!证件照换底色一分钟速成技巧

证件照的底色一般情况下分为&#xff1a;蓝色、红色、白色 证件照的尺寸也分为&#xff1a;一寸、两寸、大一寸、小二寸等 不同的文件或需求场景下对于证件照的底色和尺寸都有不同的要求&#xff0c;想要更方便的变更证件照可以先保存一版电子版&#xff0c;然后每次在初始版…

C++使用Boost库对时间的操作

0x00、获取当前时间&#xff0c;时间格式为yyyy-MM-dd hh:mm:ss.zzz std::string GetCurrentTime() {// 使用本地时间boost::posix_time::ptime now boost::posix_time::microsec_clock::local_time();// 获取毫秒部分boost::posix_time::time_duration td now.time_of_day(…

【数据结构】——双链表的实现(赋源码)

双链表的概念和结构 双链表的全称叫做&#xff1a;带头双向循环链表 它的结构示意图如下 注意&#xff1a;这⾥的“带头”跟前⾯我们说的单链表的“头结点”是两个概念&#xff0c;实际前⾯的在单链表阶段称呼不严谨&#xff0c;但是为了读者们更好的理解就直接称为单链表的头…

学习008-02-04-09 Assign a Standard Image(分配标准图像)

Assign a Standard Image&#xff08;分配标准图像&#xff09; This lesson explains how to associate an entity class with a standard image from the DevExpress.Images assembly. This image illustrates the entity class in the following sections of the UI: 本课介…

C# Unity 面向对象补全计划 之 访问修饰符

本文仅作学习笔记与交流&#xff0c;不作任何商业用途&#xff0c;作者能力有限&#xff0c;如有不足还请斧正 本系列旨在通过补全学习之后&#xff0c;给出任意类图都能实现并做到逻辑上严丝合缝

LabVIEW安装DSC模块 转自三景页593

打开NI Package Manager&#xff0c;找到LabVIEW and Drivers 找到自己需要的版本进行下载 搜索需要的模块进行下载 以DSC模块为例&#xff0c;下载右边的安装即可 最后用激活工具激活即可使用

【AI大模型】:结合wxauto实现智能微信聊天机器人

文章目录 &#x1f9d0;一、wxauto简介&#x1f3af;二、wxauto的主要功能&#x1f4e6;三、wxauto的安装与使用1. wxauto的安装2. wxauto的简单使用3. wxauto的消息对象 &#x1f4bb;四、wxauto结合大模型实现简单的聊天机器人三、完整代码 &#x1f9d0;一、wxauto简介 wxa…