Java 使用POI生成带联动下拉框的excel表格

                      java 小学生一枚 ,学习记录。

import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;

import org.apache.poi.hssf.usermodel.DVConstraint;
import org.apache.poi.hssf.usermodel.HSSFCell;
import org.apache.poi.hssf.usermodel.HSSFCellStyle;
import org.apache.poi.hssf.usermodel.HSSFDataFormat;
import org.apache.poi.hssf.usermodel.HSSFDataValidation;
import org.apache.poi.hssf.usermodel.HSSFFont;
import org.apache.poi.hssf.usermodel.HSSFRow;
import org.apache.poi.hssf.usermodel.HSSFSheet;
import org.apache.poi.hssf.usermodel.HSSFWorkbook;
import org.apache.poi.hssf.util.HSSFColor;
import org.apache.poi.ss.usermodel.DataValidation;
import org.apache.poi.ss.usermodel.Name;
import org.apache.poi.ss.util.CellRangeAddressList;

public class ExcelLinkage {

    // 样式
    private HSSFCellStyle cellStyle;

    // 初始化省份数据
    private List<String> province = new ArrayList<String>(Arrays.asList("湖南",
            "广东"));
    // 初始化数据(湖南的市区)
    private List<String> hnCity = new ArrayList<String>(Arrays.asList("长沙市",
            "邵阳市"));
    // 初始化数据(广东市区)
    private List<String> gdCity = new ArrayList<String>(Arrays.asList("深圳市",
            "广州市"));

    public void setDataCellStyles(HSSFWorkbook workbook, HSSFSheet sheet) {
        cellStyle = workbook.createCellStyle();
        // 设置边框
        cellStyle.setBorderBottom(HSSFCellStyle.BORDER_THIN);
        cellStyle.setBorderLeft(HSSFCellStyle.BORDER_THIN);
        cellStyle.setBorderRight(HSSFCellStyle.BORDER_THIN);
        cellStyle.setBorderTop(HSSFCellStyle.BORDER_THIN);
        // 设置背景色
        cellStyle.setFillForegroundColor(HSSFColor.LIGHT_GREEN.index);
        cellStyle.setFillPattern(HSSFCellStyle.SOLID_FOREGROUND);
        // 设置居中
        cellStyle.setAlignment(HSSFCellStyle.ALIGN_LEFT);
        // 设置字体
        HSSFFont font = workbook.createFont();
        font.setFontName("宋体");
        font.setFontHeightInPoints((short) 11); // 设置字体大小
        cellStyle.setFont(font);// 选择需要用到的字体格式
        // 设置单元格格式为文本格式(这里还可以设置成其他格式,可以自行百度)
        HSSFDataFormat format = workbook.createDataFormat();
        cellStyle.setDataFormat(format.getFormat("@"));
    }

    /**
     * 创建数据域(下拉联动的数据)
     *
     * @param workbook
     * @param hideSheetName
     *            数据域名称
     */
    private void creatHideSheet(HSSFWorkbook workbook, String hideSheetName) {
        // 创建数据域
        HSSFSheet sheet = workbook.createSheet(hideSheetName);
        // 用于记录行
        int rowRecord = 0;
        // 获取行(从0下标开始)
        HSSFRow provinceRow = sheet.createRow(rowRecord);
        // 创建省份数据
        this.creatRow(provinceRow, province);
        // 根据省份插入对应的市信息
        rowRecord++;
        for (int i = 0; i < province.size(); i++) {
            List<String> list = new ArrayList<String>();
            // 我这里是写死的 , 实际中应该从数据库直接获取更好
            if (province.get(i).toString().equals("湖南")) {
                // 将省份名称放在插入市的第一列, 这个在后面的名称管理中需要用到
                list.add(0, province.get(i).toString());
                list.addAll(hnCity);
            } else {
                list.add(0, province.get(i).toString());
                list.addAll(gdCity);
            }
            //获取行
            HSSFRow Cityrow = sheet.createRow(rowRecord);
            // 创建省份数据
            this.creatRow(Cityrow, list);
            rowRecord++;

        }

    }

    /**
     * 创建一列数据
     *
     * @param currentRow
     * @param textList
     */
    public void creatRow(HSSFRow currentRow, List<String> text) {
        if (text != null) {
            int i = 0;
            for (String cellValue : text) {
                // 注意列是从(1)下标开始
                HSSFCell userNameLableCell = currentRow.createCell(i++);
                userNameLableCell.setCellValue(cellValue);
            }
        }
    }

    /**
     * 名称管理
     *
     * @param workbook
     * @param hideSheetName
     *            数据域的sheet名
     */
    private void creatExcelNameList(HSSFWorkbook workbook, String hideSheetName) {
        Name name;
        name = workbook.createName();
        // 设置省名称
        name.setNameName("province");
        name.setRefersToFormula(hideSheetName + "!$A$1:$"
                + this.getcellColumnFlag(province.size())+ "$1");
        // 设置省下面的市

        for (int i = 0; i < province.size(); i++) {
            List<String> num = new ArrayList<String>();
            if (province.get(i).toString().equals("湖南")) {
                name = workbook.createName();
                num.add(0,province.get(i).toString());
                num.addAll(hnCity);
                name.setNameName(province.get(i).toString());
                name.setRefersToFormula(hideSheetName + "!$B$" + (i + 2) + ":$"
                            + this.getcellColumnFlag(num.size()) + "$" + (i + 2));
            } else {
                name = workbook.createName();
                num.add(0,province.get(i).toString());
                num.addAll(gdCity);
                name.setNameName(province.get(i).toString());
                name.setRefersToFormula(hideSheetName + "!$B$" + (i + 2) + ":$"
                            + this.getcellColumnFlag(num.size()) + "$" + (i + 2));
            }
        }
    }

    // 根据数据值确定单元格位置(比如:28-AB)
    private String getcellColumnFlag(int num) {
        String columFiled = "";
        int chuNum = 0;
        int yuNum = 0;
        if (num >= 1 && num <= 26) {
            columFiled = this.doHandle(num);
        } else {
            chuNum = num / 26;
            yuNum = num % 26;

            columFiled += this.doHandle(chuNum);
            columFiled += this.doHandle(yuNum);
        }
        return columFiled;
    }

    private String doHandle(final int num) {
        String[] charArr = { "A", "B", "C", "D", "E", "F", "G", "H", "I", "J",
                "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U", "V",
                "W", "X", "Y", "Z" };
        return charArr[num - 1].toString();
    }

    /**
     * 使用已定义的数据源方式设置一个数据验证
     *
     * @param formulaString
     * @param naturalRowIndex
     * @param naturalColumnIndex
     * @return
     */
    public DataValidation getDataValidationByFormula(String formulaString,
            int naturalRowIndex, int naturalColumnIndex) {

        // 加载下拉列表内容
        DVConstraint constraint = DVConstraint
                .createFormulaListConstraint(formulaString);
        // 设置数据有效性加载在哪个单元格上。
        // 四个参数分别是:起始行、终止行、起始列、终止列
        int firstRow = naturalRowIndex;
        int lastRow = naturalRowIndex;
        int firstCol = naturalColumnIndex - 1;
        int lastCol = naturalColumnIndex - 1;
        CellRangeAddressList regions = new CellRangeAddressList(firstRow,
                lastRow, firstCol, lastCol);
        // 数据有效性对象
        DataValidation data_validation_list = new HSSFDataValidation(regions,
                constraint);
        return data_validation_list;
    }

    /**
     * 创建一列数据
     *
     * @param hssfSheet
     */
    public void creatAppRow(HSSFSheet hssfSheet, int naturalRowIndex) {
        // 获取行
        HSSFRow hssfRow = hssfSheet.createRow(naturalRowIndex);

        HSSFCell province = hssfRow.createCell(0);
        province.setCellValue("");
        province.setCellStyle(cellStyle);

        HSSFCell City = hssfRow.createCell(1);
        City.setCellValue("");
        City.setCellStyle(cellStyle);

        // 得到验证对象
        DataValidation data_validation_list1 = this.getDataValidationByFormula(
                "province", naturalRowIndex, 1);
        DataValidation data_validation_list2 = this
                .getDataValidationByFormula("INDIRECT($A"
                        + (naturalRowIndex + 1) + ")", naturalRowIndex, 2);
        // 工作表添加验证数据
        hssfSheet.addValidationData(data_validation_list1);
        hssfSheet.addValidationData(data_validation_list2);
    }

    public void Export() {
        try {
            File file = new File("F:/excel.xls");
            FileOutputStream outputStream = new FileOutputStream(file);
            // 创建excel
            HSSFWorkbook workbook = new HSSFWorkbook();
            // 设置sheet 名称
            HSSFSheet excelSheet = workbook.createSheet("excel");
            // 设置样式
            this.setDataCellStyles(workbook, excelSheet);
            // 创建一个隐藏页和隐藏数据集
            this.creatHideSheet(workbook, "shutDataSource");
            // 设置名称数据集
            this.creatExcelNameList(workbook, "shutDataSource");
            // 创建一行数据
            for (int i = 0; i < 50; i++) {
                this.creatAppRow(excelSheet,i);

            }
            workbook.write(outputStream);
            outputStream.close();
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }

    public static void main(String[] args) {
        ExcelLinkage linkage = new ExcelLinkage();
        linkage.Export();
    }
}
时间: 2024-08-24 21:39:31

Java 使用POI生成带联动下拉框的excel表格的相关文章

使用POI生成具有三级下拉菜单的Excel文档

曾在工作中遇到这种需求,研究很久编码出一个方法,提供于此供大家参考. 效果图如下: 开始贴代码,代码中部分测试数据不影响功能. 第一部分(核心处理): 此部分包含几个方面: 获取三级下拉框各列的数据: 创建每个下拉功能的名称管理器 在隐藏的sheet中生成下拉菜单所需要的row 代码如下: /** * 第一部分 * 将三个列表所有字段从数据库查询出,并生成名称管理器,存放至隐藏的sheet中 */ private static HSSFWorkbook writePorpData() { int

导出带下拉框的Excel(数据超出长度解决方案)

  注意:此文档中标注的行号和列号都是下标,从0开始 //下载模板 protected void btnDownLoad_ServerClick(object sender,EventArgs e) { //生成Excel模板 CreateExcelTemp(); //设置想要打开的模板路径  string path = this.MapPath("~/BaseInfo/Temp/GABProdectInfoMaintain.xls"); //将此模板打开或重新保存 ExcelOutE

地址联动下拉框实现

<head> <meta charset="UTF-8"> <title>地址联动下拉框实现</title> <link rel="stylesheet" type="text/css" href="reset.css"> <link rel="stylesheet" type="text/css" href="

SharePoint 2013 使用查阅项实现联动下拉框

SharePoint列表使用中,经常会用到下拉框,而有些特殊的需求,会用到联动的下拉框,在SharePoint中默认没有这样的字段,所以如果实现,我们需要自己想办法. 这里,我们介绍如何使用JQuery+JavaScript客户端对象模型实现,下面让我简单介绍下实现的全过程. 1.创建基础列表CityList,保存的是城市名称,使用默认字段Title: 2.列表CityList的所有栏,我把Title字段的名称改为了City Name,如下图: 3.创建基础列表AreaList,用于保存所有区和

Jquery+json绑定带层次下拉框(select控件)

一.实现的效果图 二.主要代码 html代码 <select id="pid" runat="server"> <option value="0" data="|0|">不选父级类</option> </select> Jquery代码 var html = ['<option value="0">不选父级模块</option>'];

jquery+ligerform三级联动下拉框

如下为ligerform里的三级联动下拉框: var formData=[ {display:"县区",name:"QY",newline:true,labelWidth:100,width:220,space:50,type:"select",group:"区域信息",groupicon:"@Url.Content("~/Content/icons/32X32/communication.gif"

JS年月日三级联动下拉框日期选择代码

原博客网址: http://www.cnblogs.com/gdcgy/p/5467742.html 由于工作中涉及到生日编辑资料编辑,年月日用上面网址案例:bug提示: 编辑生日栏的[年]或者[月],之前保存的具体的[日]就不显示啦,产品说不管编辑哪个数据,其他数据不变: 然后自己改了一下代码: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xh

Web 1三级联动 下拉框 2添加修改删除 弹框

Web  三级联动 下拉框 using System; using System.Collections.Generic; using System.Linq; using System.Web; using System.Web.UI; using System.Web.UI.WebControls; public partial class _Default : System.Web.UI.Page { private MyDBDataContext _Context = new MyDBD

Asp.net绑定带层次下拉框(select控件)

1.效果图 2.数据库中表数据结构 3.前台页面 <select id="pid" runat="server"> <option value="0" data="|0|">不选父级类</option> </select> 备注:查看源代码 4.后台代码 using System; using System.Data; using System.Web.UI.WebControl