博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java下使用Apache POI生成具有三级联动下拉列表的Excel文档
阅读量:6160 次
发布时间:2019-06-21

本文共 8595 字,大约阅读时间需要 28 分钟。

使用Apache POI生成具有三级联动下拉列表的Excel文档;

具体效果图与代码如下文。

先上效果图:

image


开始贴代码,代码中部分测试数据不影响功能。

第一部分(核心业务处理):

此部分包含几个方面:

  1. 获取三级下拉框各列的数据;
  2. 创建每个下拉功能的名称管理器
  3. 在隐藏的sheet中生成下拉菜单所需要的row
代码如下:
/** * 第一部分 * 将三个列表所有字段从数据库查询出,并生成名称管理器,存放至隐藏的sheet中 */private static HSSFWorkbook writePorpData() {    int index = 1;    HSSFWorkbook wb = new HSSFWorkbook();   //Excel工作簿创建    wb.createSheet(DICT_SHEET_TEST);        //创建主工作表sheet    Sheet dictDataSheet = wb.createSheet(DICT_SHEET_DATA);  //创建数据源字段sheet    List
provinceList = GetData.getProvinces(); //获取所有省份 --测试数据,不影响功能 List
provinceNames = new ArrayList
(); //1.存放所有省份的名称 provinceNames.add(" "); //使下拉框有置空的选择 //遍历每个省份 for (Province province : provinceList) { String proName = province.getProvinceName(); //获取每个省份名称 provinceNames.add(proName); String provinceId = province.getProvinceId(); //获取每个省份Id List
areaList = GetData.getAreas(provinceId); //获取每个地区 --测试数据,不影响功能 List
areaNames = new ArrayList
(); //2.存放所有地区名称 areaNames.add(" "); //使下拉框有置空的选择 //遍历每个地区 for (Area area : areaList) { String areaName = area.getAreaName(); areaNames.add(areaName); String areaId = area.getAreaId(); List
cityList = GetData.getCities(areaId); //获取每个城市 --测试数据,不影响功能 List
cityNames = new ArrayList
(); //3.存放所有城市名称 cityNames.add(" "); //使下拉框有置空的选择 //遍历每个城市 for (City city : cityList) { String cityName = city.getCityName(); cityNames.add(cityName); } cityNames.add(0, areaName); createRowData(dictDataSheet.createRow(index++),cityNames);// 3.创建城市row int i2 = 0; createExcelName(wb,cityNames.get(i2++),index,cityNames.size()-1,true); //3.城市row,指定名称管理 } areaNames.add(0, proName); createRowData(dictDataSheet.createRow(index++),areaNames);// 2.创建地区row int i1 = 0; createExcelName(wb,areaNames.get(i1++),index,areaNames.size()-1,true); //2.地区row,指定名称管理 } createRowData(dictDataSheet.createRow(0),provinceNames); // 1.创建省份row,写入数据 createExcelName(wb,DICT_MNGNAME,1,provinceNames.size()-1, false); //1.省份row,指定名称管理 wb.setSheetHidden(wb.getSheetIndex(DICT_SHEET_DATA), true); //设置隐藏的sheet return wb;}

第二部分:

此部分方法都是第一部分核心处理所要使用的几个函数:

  1. 创建隐藏sheet数据行的函数
  2. 创建名称管理器的函数
  3. 创建名称管理器所需要的:计算列的表达式的函数
  4. 设置数据有效性的函数
  5. 数据验证的函数
代码如下:
/** * 第二部分:2.1     创建隐藏sheet数据行的函数 */private static void createRowData(Row curRow,List
dataList){ if(dataList != null && dataList.size()>0){ int m = 0; for (String dataValue : dataList) { Cell dataCell = curRow.createCell(m++); dataCell.setCellValue(dataValue); } }}/** * 第二部分:2.2 创建名称管理器的函数 每一行数据创建一个 */private static void createExcelName(HSSFWorkbook workbook,String nameCode,int order,int size,boolean cascadeFlag){ Name name; name = workbook.createName(); name.setNameName(nameCode); String cellString = DICT_SHEET_DATA + "!" + createExcelNameList(order,size,cascadeFlag); name.setRefersToFormula(cellString);}/** * 第二部分:2.3 名称数据行列计算表达式 */private static String createExcelNameList(int order,int size,boolean cascadeFlag){ char start='A'; if(cascadeFlag){ start = 'B'; if(size <= 25){ char end = (char)(start + size -1); return "$" + start + "$" + order + ":$" + end + "$" + order; }else{ char endPrefix = 'A'; char endSuffix = 'A'; if((size-25)/26 == 0 || size ==51){ //26-51之间,包括边界 if((size-25)%26 == 0){ //边界值 endSuffix = (char)('A' + 25); }else{ endSuffix = (char)('A' + (size-25)%26-1); } }else{ //51之上 if((size-25)%26 == 0){ endSuffix = (char)('A' + 25); endPrefix = (char)(endPrefix + (size-25)/26 -1); }else{ endSuffix = (char)('A' + (size-25)%26-1); endPrefix = (char)(endPrefix + (size-25)/26); } } return "$" + start + "$" + order + ":$" + endPrefix+endSuffix + "$" + order; } }else{ if(size<=26){ char end = (char)(start + size -1); return "$" + start + "$" + order + ":$" + end + "$" + order; }else{ char endPrefix = 'A'; char endSuffix = 'A'; if(size%26 == 0){ endSuffix = (char)('A' + 25); if(size>52 && size/26>0){ endPrefix = (char)(endPrefix + size/26-2); } }else{ endSuffix = (char)('A' + size%26-1); if(size>52 && size/26>0){ endPrefix = (char)(endPrefix + size/26-1); } } return "$" + start + "$" + order + ":$" + endPrefix+endSuffix + "$" + order; } }}/** * 第二部分:2.4 设置数据的有效性,即下拉列表的生成 */public static HSSFWorkbook getWorkbook(HSSFWorkbook wb, int size){ Sheet sheet = wb.getSheet(DICT_SHEET_TEST); DataValidation dataValidation = null; for (int x = 1; x <= size+1; x++) { dataValidation = getDataValidation("IF($B$"+x+"=\" \",\" \",INDIRECT($B$"+x+"))", x, 3); sheet.addValidationData(dataValidation); dataValidation = getDataValidation("IF($C$"+x+"=\" \",\" \",INDIRECT($C$"+x+"))", x, 4); sheet.addValidationData(dataValidation); } return wb;}/** * 第二部分:2.5 数据验证 */@SuppressWarnings("deprecation")private static DataValidation getDataValidation(String formulaString,int naturalRowIndex,int naturalColIndex){ //设置数据有效性加载在哪个单元格上 四个参数:起始行、终止行、起始列、终止列 int firstRow = naturalRowIndex-1; int lastRow = naturalRowIndex-1; int firstCol = naturalColIndex-1; int lastCol = naturalColIndex-1; CellRangeAddressList regions = new CellRangeAddressList(firstRow, lastRow, firstCol, lastCol); //加载下拉列表 DVConstraint constraint = DVConstraint.createFormulaListConstraint(formulaString); //数据有效性对象 DataValidation dataValidation = new HSSFDataValidation(regions, constraint); //设置输入信息提示信息 dataValidation.createPromptBox("下拉提示", "请选择合适的值"); //设置输入错误提示信息 dataValidation.createErrorBox("非法输入", "不允许输入,请选取下拉值!"); return dataValidation;}

第三部分:

此部分即获得上两部处理完成后的工作簿,然后填充数据即可。

代码如下:
/** * 创建并生成excel文档 */public static void createExcelFile(){    List
studens = GetData.getStudents(); //测试数据,不影响功能 try { FileOutputStream fileOutputStream = new FileOutputStream(new File(filePathName)); HSSFWorkbook wb = writePorpData(); // 创建工作簿 HSSFSheet sheet = wb.getSheet(DICT_SHEET_TEST); // 获取主工作表 wb = getWorkbook(wb, studens.size()); HSSFRow row = null; HSSFCell cell = null; sheet.setDefaultColumnWidth(28); row = sheet.createRow(0); // 新增标题行 cell = row.createCell(0); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue("学生"); cell = row.createCell(1); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue("省份"); cell = row.createCell(2); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue("地区"); cell = row.createCell(3); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue("城市"); int i = 1; for (Student stu : studens) { row = sheet.createRow(i); // 新增一行 cell = row.createCell(0); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue(stu.getStudentName()); // 学生姓名 cell = row.createCell(1); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue(stu.getProvince()); //省份 cell = row.createCell(2); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue(stu.getArea()); // 地区 cell = row.createCell(3); cell.setCellType(HSSFCell.CELL_TYPE_STRING); cell.setCellValue(stu.getCity()); //城市 i++; } sheet.setColumnWidth(0,5000);//设置列宽 sheet.setColumnWidth(1,5000);//设置列宽 sheet.setColumnWidth(2,5000);//设置列宽 sheet.setColumnWidth(3,5000);//设置列宽 wb.write(fileOutputStream); //生成文档 fileOutputStream.close(); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); }}

  • 附上源码下载:

转载于:https://www.cnblogs.com/dcncy/p/8153191.html

你可能感兴趣的文章
3springboot:springboot配置文件(外部配置加载顺序、自动配置原理,@Conditional)
查看>>
图解SSH原理及两种登录方法
查看>>
查询个人站点的文章、分类和标签查询
查看>>
基础知识:数字、字符串、列表 的类型及内置方法
查看>>
JSP的隐式对象
查看>>
JS图片跟着鼠标跑效果
查看>>
Leetcode 3. Longest Substring Without Repeating Characters
查看>>
416. Partition Equal Subset Sum
查看>>
app内部H5测试点总结
查看>>
[TC13761]Mutalisk
查看>>
while()
查看>>
常用限制input的方法
查看>>
IIS7下使用urlrewriter.dll配置
查看>>
并行程序设计学习心得1——并行计算机存储
查看>>
C++ 迭代器运算
查看>>
【支持iOS11】UITableView左滑删除自定义 - 实现多选项并使用自定义图片
查看>>
【算法笔记】多线程斐波那契数列
查看>>
java8函数式编程实例
查看>>
jqgrid滚动条宽度/列显示不全问题
查看>>
在mac OS10.10下安装 cocoapods遇到的一些问题
查看>>