Java下使用Apache POI生成具有三级联动下拉列表的Excel文档
使用Apache POI生成具有三级联动下拉列表的Excel文档;
具体效果图与代码如下文。先上效果图:

开始贴代码,代码中部分测试数据不影响功能。
第一部分(核心业务处理):
此部分包含几个方面:
- 获取三级下拉框各列的数据;
- 创建每个下拉功能的名称管理器
- 在隐藏的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<Province> provinceList = GetData.getProvinces(); //获取所有省份 --测试数据,不影响功能
List<String> provinceNames = new ArrayList<String>(); //1.存放所有省份的名称
provinceNames.add(" "); //使下拉框有置空的选择
//遍历每个省份
for (Province province : provinceList) {
String proName = province.getProvinceName(); //获取每个省份名称
provinceNames.add(proName);
String provinceId = province.getProvinceId(); //获取每个省份Id
List<Area> areaList = GetData.getAreas(provinceId); //获取每个地区 --测试数据,不影响功能
List<String> areaNames = new ArrayList<String>(); //2.存放所有地区名称
areaNames.add(" "); //使下拉框有置空的选择
//遍历每个地区
for (Area area : areaList) {
String areaName = area.getAreaName();
areaNames.add(areaName);
String areaId = area.getAreaId();
List<City> cityList = GetData.getCities(areaId); //获取每个城市 --测试数据,不影响功能
List<String> cityNames = new ArrayList<String>(); //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;
}
第二部分:
此部分方法都是第一部分核心处理所要使用的几个函数:
- 创建隐藏sheet数据行的函数
- 创建名称管理器的函数
- 创建名称管理器所需要的:计算列的表达式的函数
- 设置数据有效性的函数
- 数据验证的函数
代码如下:
/**
* 第二部分:2.1 创建隐藏sheet数据行的函数
*/
private static void createRowData(Row curRow,List<String> 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<Student> 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();
}
}
- 附上源码下载:源码
Java下使用Apache POI生成具有三级联动下拉列表的Excel文档的更多相关文章
- 【Java】使用Apache POI生成和解析Excel文件
概述 Excel是我们平时工作中比较常用的用于存储二维表数据的,JAVA也可以直接对Excel进行操作,分别有jxl和poi,2种方式. HSSF is the POI Project's pure ...
- python3如何随机生成大数据存储到指定excel文档里
本次主要采用的是python3的第三方库xlwt,来创建一个excel文件.具体步骤如下: 1.确认存储位置,文件命名跟随时间格式 2.封装写入格式 3.实现随机数列生成 4.定位行和列把随机数写入 ...
- java用org.apache.poi包操作excel
一.POI简介 Jakarta POI 是apache的子项目,目标是处理ole2对象.它提供了一组操纵Windows文档的Java API 目前比较成熟的是HSSF接口,处理MS Excel(97- ...
- Java使用poi包读取Excel文档
项目需要解析Excel文档获取数据,就在网上找了一些资料,结合自己这次使用,写下心得: 1.maven项目需加入如下依赖: <dependency> <groupId>org. ...
- Java之Poi导出Excel文档
一.Poi简介 在后台管理系统中,我们经常要做的导出操作,通常导出为Excel文档的形式,而Poi则提供了这种需要的支持. 二.Workbook/HSSFWorkbook/XSSFWorkbook 1 ...
- java、ruby、python、php等如何生成excel文档?
excel在我们日常工作生活中会经常用到,通常我们都是用office软件去编写文档.但是对于格式一致的excel文档,如果还是使用人工完成,那绝不是我们软件工程师的姿态了~ 下面我就介绍一种方法,不需 ...
- java生成excel文档
要做一个后台自动化,要先预先生成一份文档,以下内容生成了文档 首先下载jxl.jar包,下载地址:http://download.csdn.net/detail/prstaxy/4469935 1.生 ...
- struts2中利用POI导出Excel文档并下载
1.项目组负责人让我实现这个接口,因为以前做过类似的,中间并没有遇到什么太困难的事情.其他不说,先上代码: package com.tydic.eshop.action.feedback; impor ...
- POI 读取Excel文档中的数据——兼容Excel2003和Excel2007
Apache POI是Apache软件基金会的开放源码函式库,POI提供API给Java程序对Microsoft Office格式档案读和写的功能. HSSF - 提供读写Microsoft Exce ...
随机推荐
- iOS10 相册权限
当我升级到Xcode8后,启动我的相机项目,直接crash,输出的日志如下: '2016-07-08 16:41:11.268943 project-name[362:56625] [MC] Syst ...
- 图片验证码(Struts2中使用)
写在前面: 最近在项目中做了一个登录页面,用到了图片验证码的功能,所以记录一下.方便之后再有用到,直接拿来用即可.其实图片验证码的生成都是有固定步骤的,网上也有很多的例子,有的时候,如果不想深究,都是 ...
- 【python】for循环
>>> exp='welcom to python'>>> for i in exp: print(i,end=' ') w e l c o m t o p y t ...
- IOS学习1——IOS应用程序的生命周期及基本架构
一.应用程序的状态和多任务 有时系统会从app一种状态切换另一种状态来响应系统发生的事件.例如,当用户按下home键.电话打入.或其他中断发生时,当前运行的应用程序会切换状态来响应.应用程序的状态有以 ...
- Android 环境搭建、基础窗口window/Mac
1.五步搞定Android开发环境部署--非常详细的Android开发环境搭建教程 2.Android开发学习之路--MAC下Android Studio开发环境搭建 4.Android常用开发工具以 ...
- JaveScript流程控制(JS知识点归纳四)
01 流程控制 顺序结构: 程序的默认执行方式 条件判断语句:也称之为分支结构,选择结构:如果程序要执行的代码出现了多种情况需要使用 循环结构:当代码需要多次重复执行多次时,使用 02 条件判断语句 ...
- Jmeter3.2版本中Generating Report Dashboard功能浅析
自从投入到Jmeter怀抱,一直想找到一个比较不错的测试结果报告模板用于展示,类似于Loadrunner中导出html那种,但是苦苦的寻找始终没有一款让我看上眼的,包括Jmeter自带的xsl,虽然展 ...
- JDK8中JVM对类的初始化探讨
在<深入理解Java虚拟机>(第二版,周志明著)中,作者介绍了JVM必须初始化类(或接口)的五种情况,但是是针对JDK7而言的. 那么,在JDK8中,这几种情况有没有变化呢?(我猜测应该会 ...
- c#程序连接mysql,报"Illegal mix of collations (utf8_general_ci,IMPLICIT) and (utf8_unicode_ci,IMPLICIT) for operation '='"的解决方案
=============================================== 20170607_第一次修改 ccb_warlock === ...
- Jasperreports以及iReport4.5报表PDF导出字体完美解决方案
在使用Jasperreports以及iReport设计报表时,导出PDF是一个常见的需求.网上解决PDF导出中文显示问题相关的文章很多,无非就是设置控件的pdf font name和pdf encod ...