POI生成Excel文件增加数据验证(下拉序列)

POI版本为5.2.2
正常的如果不超过255字符的数据验证可以参照如下代码:

java 复制代码
/**
 * <p>设置某列的数据验证</p>
 * @param Sheet 作用于哪一个sheet
 * @param colIndex 需要增加数据验证的列的索引
 * @String[] names 数据验证的序列,就是excel下拉列表的内容
 */
public static void setDataValidation(Sheet sheet, int colIndex, String[] names) {
    //直接用给定的数据创建显式序列
	DataValidationConstraint constraint = sheet.getDataValidationHelper().createExplicitListConstraint(names);
	constraint.setOperator(DataValidationConstraint.OperatorType.BETWEEN);
    //创建一个区域,某列的第2行到第65535行,一般第一行是标题列
	CellRangeAddressList addressList = new CellRangeAddressList(1,65535,colIndex,colIndex);
	DataValidation validation = sheet.getDataValidationHelper().createValidation(constraint, addressList);
	validation.setErrorStyle(DataValidation.ErrorStyle.STOP);
    //不允许单元格为空
	validation.setEmptyCellAllowed(false);
	validation.setShowErrorBox(true);
    //添加数据验证到sheet中
	sheet.addValidationData(validation);
}

@Test
public void test(){
    Workbook workbook = WorkbookFactory.create(true);//xlsx
    Sheet sheet = workbook.createSheet("测试");
    //先创建65535行
    for (int i = 0; i < 65535; i++) {
        Row row = sheet.getRow(i);
        if (Objects.isNull(row))
            row=sheet.createRow(i);
        Cell cell = row.getCell(0);
        if (Objects.isNull(cell))
            cell = row.createCell(0);
    }
    //下拉列表的数据
    String[] strs = {"借","贷"};
    setDataValidation(sheet,0,strs);
    workbook.write(new FileOutputStream("D:\\poitest\\测试.xlsx"));
}

由于excel数据验证 序列的长度最多255字符, 如果代码中数据的长度超过这个,则生成的excel会提示先修复才能打开,修复完成打开之后发现需要设置数据验证的列没有设置成功。如果想要输入更多数据,则先可以先把数据保存到一个隐藏的sheet中,然后通过公式引用存储的数据再设置数据验证。

java 复制代码
/**
 * <p>设置某列的数据验证</p>
 * @param Sheet 作用于哪一个sheet
 * @param colIndex 需要增加数据验证的列的索引
 * @String formula 公式,引用隐藏sheet的数据
 */
public static void setDataValidationByFormula(Sheet sheet, int colIndex, String formula) {
    //直接用给定的公式创建
	DataValidationConstraint constraint = sheet.getDataValidationHelper().createFormulaListConstraint(formula);
	constraint.setOperator(DataValidationConstraint.OperatorType.BETWEEN);
    //创建一个区域,某列的第2行到第65535行,一般第一列是标题列
	CellRangeAddressList addressList = new CellRangeAddressList(1,65535,colIndex,colIndex);
	DataValidation validation = sheet.getDataValidationHelper().createValidation(constraint, addressList);
	validation.setErrorStyle(DataValidation.ErrorStyle.STOP);
    //不允许单元格为空
	validation.setEmptyCellAllowed(false);
	validation.setShowErrorBox(true);
    //添加数据验证到sheet中
	sheet.addValidationData(validation);
}

@Test
public void test(){
    Workbook workbook = WorkbookFactory.create(true);//xlsx
    Sheet sheet = workbook.createSheet("测试");
    //先创建65535行
    for (int i = 0; i < 65535; i++) {
        Row row = sheet.getRow(i);
        if (Objects.isNull(row))
            row=sheet.createRow(i);
        Cell cell = row.getCell(0);
        if (Objects.isNull(cell))
            cell = row.createCell(0);
    }
    //下拉列表的数据,一般从数据库取出
    List<String> acctitleList = getAcctitleList();
    Sheet hiddenSheet = workbook.createSheet("hidden");
    for (int i = 0; i < acctitleList.size(); i++) {
        Row row = hiddenSheet.createRow(i);
        Cell cell = row.createCell(0);
        cell.setCellValue(acctitleList.get(i));
    }
    //hidden 的索引为1,隐藏sheet
    workbook.setSheetHidden(1,true);
    //格式为  sheet名字!$起始列名$起始行号:$结束列名$结束行号
    String formula = "hidden!$A$1:$A$"+acctitleList.size();
    setDataValidation(sheet,0,formula);
    workbook.write(new FileOutputStream("D:\\poitest\\测试.xlsx"));
}
/**
 * 模拟从数据库中取数据
 */
private List<String> getAcctitleList(){
    List<String> list = new ArrayList<>();
    list.add("6405020000 | 税金及附加-土地增值税");
    list.add("6405090000 | 税金及附加-城市维护建设税");
    list.add("6405100000 | 税金及附加-教育费附加");
    list.add("6405030000 | 税金及附加-房产税");
    list.add("6405040000 | 税金及附加-土地使用税");
    list.add("6405110000 | 税金及附加-地方教育附加");
    list.add("6405080000 | 税金及附加-印花税");
    list.add("6405120000 | 税金及附加-水利建设基金");
    list.add("2221210000 | 应交税费-应交城市维护建设税");
    list.add("2221140000 | 应交税费-应交土地增值税");
    list.add("2221220000 | 应交税费-应交教育费附加");
    list.add("2221150000 | 应交税费-应交房产税");
    list.add("2221160000 | 应交税费-应交土地使用税");
    list.add("2221230000 | 应交税费-应交地方教育费附加");
    list.add("2221200000 | 应交税费-应交印花税");
    list.add("2221240000 | 应交税费-水利建设基金");
    list.add("5005040400 | 服务成本-劳务费-保绿费");
    list.add("2241050000 | 其他应付款-暂估/预提费");
    return list;
}
相关推荐
V+zmm101342 分钟前
基于微信小程序的乡村政务服务系统springboot+论文源码调试讲解
java·微信小程序·小程序·毕业设计·ssm
Oneforlove_twoforjob27 分钟前
【Java基础面试题025】什么是Java的Integer缓存池?
java·开发语言·缓存
xmh-sxh-131429 分钟前
常用的缓存技术都有哪些
java
AiFlutter1 小时前
Flutter-底部分享弹窗(showModalBottomSheet)
java·前端·flutter
J不A秃V头A1 小时前
IntelliJ IDEA中设置激活的profile
java·intellij-idea
DARLING Zero two♡2 小时前
【优选算法】Pointer-Slice:双指针的算法切片(下)
java·数据结构·c++·算法·leetcode
小池先生2 小时前
springboot启动不了 因一个spring-boot-starter-web底下的tomcat-embed-core依赖丢失
java·spring boot·后端
CodeClimb2 小时前
【华为OD-E卷-木板 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od
程序员厉飞雨2 小时前
Android R8 耗时优化
android·java·前端
odng2 小时前
IDEA自己常用的几个快捷方式(自己的习惯)
java·ide·intellij-idea