POI 之 XSSFWorkBook 加载复杂表头,填充数据,并实现导出

项目场景:报表导出,复杂表头

需求:项目需要导出一份报表,表头比较复杂。表头存在三行,并且各种组合。

在这里插入图片描述


问题分析:虽然复杂,但是静态表头

说明:需要的报表表头虽然复杂,好在不是动态表头,上面的列名是固定的,产生读取模板后填充数据的想法。

根据提供的样表,得到excel模板:“消耗品.xlsx”
在这里插入图片描述


解决方案:

首先明确一下基本概念:
  先创建一个工作簿,一个工作簿可以有多个工作表,一个工作表可以有多个行,一个行可以有多个单元格

  • 工作簿 ----------->XSSFWorkbook
  • 工作表 ----------->XSSFSheet
  • 行 ----------->XSSFRow
  • 单元格 ----------->XSSFCell

关键代码拆分:

第一步:从模板表加载复杂表头。

		//测试2:加载模板的表头
        string TempletFileName = Server.MapPath("~/DownloadFile/消耗品.xlsx");
        FileStream file = new FileStream(TempletFileName, FileMode.Open, FileAccess.Read);
        XSSFWorkbook workbook = new XSSFWorkbook(file);
        XSSFSheet sheet = (XSSFSheet)workbook.GetSheet("Sheet1");

第二步:往表内写入数据

 //往表内写入数据
        for (int i = 0; i < dt.Rows.Count; i++)
        {  //数据行数 循环
            IRow row = sheet.CreateRow(i + 3);    //创建 行 的起始行 +3表示表头占用了3行
            //模板表有35列
            for (int j = 0; j < 35; j++)
            {
                //创建单元格
                ICell cells = row.CreateCell(j);
                //为单元格赋值
                cells.SetCellValue(dt.Rows[i][j].ToString());
                //列宽自适应
                sheet.AutoSizeColumn(j);
            }
        }

第三步:表格导出

//将表内容写入流 通知浏览器下载
        MemoryStream ms = new MemoryStream();
        workbook.Write(ms);
        System.Web.HttpContext.Current.Response.AddHeader("Content-Disposition", string.Format("attachment;filename={0}.xlsx", fileName));
        System.Web.HttpContext.Current.Response.AddHeader("Content-Length", ms.ToArray().Length.ToString());
        System.Web.HttpContext.Current.Response.BinaryWrite(ms.ToArray()); //进行二进制流下载

        workbook = null;
        ms.Close();
        ms.Dispose();

        System.Web.HttpContext.Current.Response.End();

总结:完整Demo


public void ExportExcel(DataTable dt, string fileName)
    {
        //测试2:加载模板的表头
        string TempletFileName = Server.MapPath("~/DownloadFile/污水表.xlsx");
        FileStream file = new FileStream(TempletFileName, FileMode.Open, FileAccess.Read);
        XSSFWorkbook workbook = new XSSFWorkbook(file);
        XSSFSheet sheet = (XSSFSheet)workbook.GetSheet("Sheet1");
        
        //设置单元格样式:水平、垂直居中
        ICellStyle titlestyle = workbook.CreateCellStyle();
        titlestyle.Alignment = HorizontalAlignment.Center;
        titlestyle.VerticalAlignment = VerticalAlignment.Center;


        //往表内写入数据
        for (int i = 0; i < dt.Rows.Count; i++)
        {  //数据行数 循环
            IRow row = sheet.CreateRow(i + 3);    //模板表表头占用3行,所以从第四行开始填充数据
            //模板表有35列
            for (int j = 0; j < 35; j++)
            {
                ICell cells = row.CreateCell(j);
                cells.CellStyle = titlestyle;
                //按单元格类型 赋值
                if (0 < j && j < 33){
                    //整数型 赋值办法
                    cells.SetCellValue(Convert.ToDouble(dt.Rows[i][j].ToString().Length == 0 ? null : (dt.Rows[i][j])));
                }
                //第一列和最后两列内容是 :字符串
                else{
                	//字符型 赋值办法
                    cells.SetCellValue(dt.Rows[i][j].ToString());
                    sheet.AutoSizeColumn(j);//采用自适应列宽
                }
            }
        }

        //将表内容写入流 通知浏览器下载
        MemoryStream ms = new MemoryStream();
        workbook.Write(ms);
        System.Web.HttpContext.Current.Response.AddHeader("Content-Disposition", string.Format("attachment;filename={0}.xlsx", fileName));
        System.Web.HttpContext.Current.Response.AddHeader("Content-Length", ms.ToArray().Length.ToString());
        System.Web.HttpContext.Current.Response.BinaryWrite(ms.ToArray()); //进行二进制流下载

        workbook = null;
        ms.Close();
        ms.Dispose();
//这句关闭流,不加的话,下载的文件会需要修复才能打开
        System.Web.HttpContext.Current.Response.End();
    }
  • 2
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
好的,关于Java导出excel实现一对多复杂表头导出代码,以下是一个简单的示例代码: 1. 首先,需要创建一个POJO类来存储表格数据,例如ExcelData.java: ``` public class ExcelData { private String headerName; private List<String> columnNames; private List<List<String>> tableData; // 构造函数,getter和setter方法省略 } ``` 2. 为了实现一对多复杂表头,我们需要使用POI库来操作Excel文件,可以先在pom.xml文件中引入POI依赖: ``` <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi</artifactId> <version>4.1.2</version> </dependency> ``` 3. 接着,可以编写导出Excel的方法: ``` public static void exportExcel(List<ExcelData> dataList, String fileName, HttpServletResponse response) { // 设置响应头信息 response.setContentType("application/octet-stream"); response.setHeader("Content-Disposition", "attachment;filename=" + fileName); // 创建Workbook对象 Workbook workbook = new XSSFWorkbook(); // 遍历数据列表 for (ExcelData data : dataList) { // 创建Sheet对象 Sheet sheet = workbook.createSheet(data.getHeaderName()); // 创建表头行 Row headerRow = sheet.createRow(0); // 创建表头单元格 Cell headerCell = headerRow.createCell(0); headerCell.setCellValue(data.getHeaderName()); sheet.addMergedRegion(new CellRangeAddress(0, 1, 0, data.getColumnNames().size() - 1)); // 创建子表头行 Row subHeaderRow = sheet.createRow(1); // 创建子表头单元格 int columnIndex = 0; for (String columnName : data.getColumnNames()) { Cell subHeaderCell = subHeaderRow.createCell(columnIndex++); subHeaderCell.setCellValue(columnName); } // 填充数据 int rowIndex = 2; for (List<String> rowData : data.getTableData()) { Row dataRow = sheet.createRow(rowIndex++); columnIndex = 0; for (String columnValue : rowData) { Cell dataCell = dataRow.createCell(columnIndex++); dataCell.setCellValue(columnValue); } } // 调整列宽 for (int i = 0; i < data.getColumnNames().size(); i++) { sheet.autoSizeColumn(i); } } // 输出Excel文件 try { OutputStream outputStream = response.getOutputStream(); workbook.write(outputStream); workbook.close(); outputStream.flush(); outputStream.close(); } catch (IOException e) { e.printStackTrace(); } } ``` 4. 最后,在Controller中调用导出Excel的方法即可: ``` @RequestMapping("/exportExcel") public void exportExcel(HttpServletResponse response) { List<ExcelData> dataList = getDataList(); // 从数据库中获取数据列表 String fileName = "test.xlsx"; // Excel文件的名称 ExcelUtil.exportExcel(dataList, fileName, response); // 导出Excel文件 } ``` 以上就是Java导出excel实现一对多复杂表头导出代码的示例。希望能对你有所帮助!

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值