Java EasyExcel 支持动态表头和单元格下拉导出

该博客介绍了如何使用EasyExcel库动态创建Excel模板并设置单元格的下拉选项。通过测试类`DynamicWriteTest`,展示了如何准备表头、下拉列表数据,并利用自定义拦截器`DynamicSheetWriteHandler`实现数据验证和下拉框功能。测试方法中创建了一个带有动态表头的Excel文件,并设置了特定列的下拉选项,实现了动态导出模板的功能。

需求背景:对于有些表头需要根据配置动态生成,做以下记录。

动态导出模板

测试工具类

package com.alibaba.easyexcel.test.demo.write;

import com.alibaba.easyexcel.test.util.TestFileUtil;
import com.alibaba.excel.EasyExcel;
import com.alibaba.excel.util.ListUtils;
import org.junit.Test;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.stream.Collectors;

/**
 * @description:
 * @author: root
 * @date: 2022-11-16
 */
public class DynamicWriteTest {

    /**
     * 测试方法
     * @param
     * @return
     * @author root
     * @date 2022-11-16
     */
    @Test
    public void writeDynamicHead() {
        // 表头
        List<List<String>> headers = prepareHeaders();
        // 单元格下拉
        HashMap<Integer, List<String>> dropCells = prepareDropCells();

        // 模板导出
        String fileName = TestFileUtil.getPath() + "customHandlerWrite" + System.currentTimeMillis() + ".xlsx";
        EasyExcel.write(fileName).head(headers).registerWriteHandler(new DynamicSheetWriteHandler(dropCells)).sheet("模板").doWrite(new ArrayList<>());
    }

    /**
     * 测试数据
     * @param
     * @return {@link List< DemoData>}
     * @author root
     * @date 2022-11-16
     */
    private List<DemoData> data() {
        List<DemoData> list = ListUtils.newArrayList();
        for (int i = 0; i < 10; i++) {
            DemoData data = new DemoData();
            data.setString("字符串" + i);
            data.setDate(new Date());
            data.setDoubleData(0.56);
            list.add(data);
        }
        return list;
    }

    private static final List<String> headers = Arrays.asList("动态表头1", "动态表头2", "动态表头3");
    private static final List<String> dropCells = Arrays.asList("测试1", "测试2");

    /**
     * 准备动态表头数据
     * @param
     * @return {@link List< List< String>>}
     * @author root
     * @date 2022-11-16
     */
    private List<List<String>> prepareHeaders() {
        List<List<String>> list = headers.stream().map(item -> Arrays.asList(item)).collect(Collectors.toList());
        return list;
    }

    /**
     * 准备单元格下拉数据
     * @param
     * @return {@link HashMap< Integer, List< String>>}
     * @author root
     * @date 2022-11-16
     */
    private HashMap<Integer, List<String>> prepareDropCells() {
        HashMap<Integer, List<String>> dropCellMap = new HashMap<>();

        for(int i = 0; i < headers.size(); i++) {
            if(headers.get(i).equals("动态表头1")) {
                dropCellMap.put(i, dropCells);
            }
        }
        return dropCellMap;
    }

}

自定义拦截器

package com.alibaba.easyexcel.test.demo.write;

import com.alibaba.excel.write.handler.SheetWriteHandler;
import com.alibaba.excel.write.handler.context.SheetWriteHandlerContext;
import lombok.extern.slf4j.Slf4j;
import org.apache.poi.hssf.usermodel.HSSFDataValidation;
import org.apache.poi.ss.usermodel.DataValidation;
import org.apache.poi.ss.usermodel.DataValidationConstraint;
import org.apache.poi.ss.usermodel.DataValidationHelper;
import org.apache.poi.ss.util.CellRangeAddressList;

import java.util.HashMap;
import java.util.List;

/**
 * 自定义拦截器. 新增单元格下拉框数据
 *
 * @author root
 */
@Slf4j
public class DynamicSheetWriteHandler implements SheetWriteHandler {

    private HashMap<Integer, List<String>> dropCellMap;

    public DynamicSheetWriteHandler(HashMap<Integer, List<String>> dropCellMap) {
        this.dropCellMap = dropCellMap;
    }

    @Override
    public void afterSheetCreate(SheetWriteHandlerContext context) {
        log.info("第{}个Sheet写入成功。", context.getWriteSheetHolder().getSheetNo());

        dropCellMap.forEach((k, v) -> {
            CellRangeAddressList cellRangeAddressList = new CellRangeAddressList(1, 65535, k, k);
            DataValidationHelper helper = context.getWriteSheetHolder().getSheet().getDataValidationHelper();
            DataValidationConstraint constraint = helper.createExplicitListConstraint(v.toArray(new String[0]));
            DataValidation dataValidation = helper.createValidation(constraint, cellRangeAddressList);

            //设置约束
            if(dataValidation instanceof HSSFDataValidation) {
                dataValidation.setSuppressDropDownArrow(false);
            } else {
                dataValidation.setSuppressDropDownArrow(true);
                dataValidation.setShowErrorBox(true);
            }
            // 禁止输入非下拉框的内容
            dataValidation.setErrorStyle(DataValidation.ErrorStyle.STOP);
            dataValidation.createErrorBox("提示", "此值与单元格下拉不符");
            context.getWriteSheetHolder().getSheet().addValidationData(dataValidation);
        });

    }
}

结果展示:

 

### EasyExcel 导出多级表头的使用指南 在实际业务中,导出具有多级表头的 Excel 文件是常见的需求,尤其是在报表展示中。EasyExcel 提供了便捷的方式来实现多级表头的导出。以下将详细介绍如何使用 EasyExcel 实现多级表头的导出。 #### 1. 使用 `List<List<String>>` 构建动态表头 通过构建 `List<List<String>>` 类型的表头集合,可以实现多级表头的定义。每个子列表代表一行表头,其中每个元素代表该层级的一个表头单元格。这种方式适用于动态构建表头的情况。 ```java List<List<String>> headList = new ArrayList<>(); // 一级表头 headList.add(Arrays.asList("一级1", "一级2")); // 二级表头 headList.add(Arrays.asList("二级1-1", "二级1-2", "二级2-1", "二级2-2")); // 三级表头 headList.add(Arrays.asList("三级1-1-1", "三级1-1-2", "三级1-2-1", "三级1-2-2", "三级2-1-1", "三级2-1-2", "三级2-2-1", "三级2-2-2")); ``` 这种方式可以灵活应对不同层级的表头需求,并且可以结合数据动态生成[^1]。 #### 2. 使用实体类注解方式定义多级表头 通过 `@ExcelProperty` 注解,可以为实体类的字段定义多级表头。每个字段的注解参数是一个字符串数组,表示该字段在表头中的路径。 ```java @Data public class ExportItem { @ExcelProperty({"名称"}) private String name; @ExcelProperty({"一级", "二级头"}) private String data1; @ExcelProperty({"一级", "二级头", "三级头", "四级"}) private String data2; @ExcelProperty({"一级", "二级头", "三级头", "四级"}) private String data3; @ExcelIgnore private String ignoreData; } ``` 这种方式适合结构较为固定的表头导出,能够与数据模型紧密结合,便于数据绑定导出[^2]。 #### 3. 导出时设置样式与列宽 在导出时,可以通过自定义样式类来设置表头或数据单元格的样式,例如字体、背景颜色、对齐方式等。同时,也可以设置列宽以适应不同内容。 ```java // 定义样式 WriteCellStyle headStyle = new WriteCellStyle(); headStyle.setFillForegroundColor(IndexedColors.GREY_25_PERCENT.getIndex()); WriteFont headFont = new WriteFont(); headFont.setFontHeightInPoints((short) 12); headFont.setBold(true); headStyle.setWriteFont(headFont); // 导出时应用样式 EasyExcel.write(filePath) .head(headList) .registerWriteHandler(new SimpleColumnWidthStrategy(20)) .registerWriteHandler(new WriteHandlerChainModel(Arrays.asList(headStyle))) .sheet("Sheet1") .doWrite(dataList); ``` 通过上述方式,可以实现表头数据的样式定制,提升导出文件的可读性[^1]。 #### 4. 导出多表头并包含下拉框 在导出多级表头的同时,还可以为某些列设置下拉选框,提升数据输入的规范性。可以通过 `EasyExcelUtil` 工具类实现这一功能。 ```java @GetMapping("/exportSelectedExcelDemo") public void exportDemo(HttpServletRequest request, HttpServletResponse response) { String[] arr = {"男", "女"}; String[] group = {"蓝队", "绿队"}; HashMap<String, List<String>> stringHashMap = new HashMap<>(); stringHashMap.put("sex", Arrays.asList(arr)); stringHashMap.put("group", Arrays.asList(group)); EasyExcelUtil.generatorTemplateBySelected(response, null, stringHashMap, User.class); } ``` 此方法可以在导出模板时,为特定列添加下拉选项,方便后续数据录入[^3]。 #### 5. Maven 依赖配置 在项目中使用 EasyExcel,需要引入相应的 Maven 依赖: ```xml <!-- EasyExcel --> <dependency> <groupId>com.alibaba</groupId> <artifactId>easyexcel</artifactId> <version>3.2.1</version> </dependency> ``` 确保依赖版本与项目兼容,并正确配置构建工具(如 Maven 或 Gradle)[^4]。 ---
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值