Java实现导出word表格的示例详解
作者:步尔斯特 发布时间:2023-01-02 21:36:48
标签:Java,导出,word,表格
目标
多级表头、分页、动态数据
实现
依赖
<!-- poi工具类-->
<dependency>
<groupId>com.deepoove</groupId>
<artifactId>poi-tl</artifactId>
<version>1.12.0</version>
</dependency>
模版
代码
TableData数据(模版对应的数据对象)
package org.example.bean;
import com.deepoove.poi.data.TableRenderData;
import lombok.Data;
@Data
public class TableData {
/**
* 标题
*/
private String title;
/**
* 表格
*/
private TableRenderData table;
private String[][] tableList;
/**
* 总价
*/
private String totalPrice;
}
核心代码
package org.example.controller;
import lombok.SneakyThrows;
import org.example.bean.TableData;
import org.springframework.web.bind.annotation.*;
import com.deepoove.poi.XWPFTemplate;
import com.deepoove.poi.data.*;
import java.io.*;
import java.math.BigDecimal;
import java.net.URLEncoder;
import javax.servlet.http.HttpServletResponse;
/**
* Java导出word表格
* 根据word模版,手绘表格
*/
@RestController
@RequestMapping(value = "/word")
public class WordController {
@GetMapping(value = "/table")
@SneakyThrows
public void table(TableData tableData, HttpServletResponse response) {
/* 假数据 */
tableData.setTitle("附件1-报价明细表");
String[][] strings = new String[100][5];
for (int i = 0; i < 100; i++) {
strings[i] = new String[]{"1", "EREWHON", "生猪", "酒鬼酒", "125"};
}
tableData.setTableList(strings);
// 模版路径
String wordPath = "/Users/issavior/java/java/seckill-redis/test/src/main/resources/";
String modelName = "表格.docx";
// 手绘表格
// 表头
RowRenderData row0 = Rows.of("项号", "编号", "种类", "", "价格").center().create();
RowRenderData row1 = Rows.of("项号", "编号", "期货", "股票", "价格").center().create();
int length = 0;
if (tableData.getTableList() != null) {
length = tableData.getTableList().length;
}
// 表格数据 加上2行表头 再加上最后一行总价
RowRenderData[] rowRenderData = new RowRenderData[length + 3];
rowRenderData[0] = row0;
rowRenderData[1] = row1;
// 计算价钱
BigDecimal totalPrice = new BigDecimal("0");
for (int i = 0; i < length; i++) {
rowRenderData[i + 2] = Rows.of(tableData.getTableList()[i]).center().create();
String s = tableData.getTableList()[i][4];
BigDecimal bigDecimal = new BigDecimal(s);
totalPrice = totalPrice.add(bigDecimal);
}
RowRenderData row4 = Rows.of("总价", "", "", "", totalPrice.toString()).center().create();
rowRenderData[rowRenderData.length - 1] = row4;
// 表格合并,根据坐标
MergeCellRule rule = MergeCellRule.builder().map(MergeCellRule.Grid.of(0, 0), MergeCellRule.Grid.of(1, 0)).
map(MergeCellRule.Grid.of(0, 1), MergeCellRule.Grid.of(1, 1)).
map(MergeCellRule.Grid.of(0, 2), MergeCellRule.Grid.of(0, 3)).
map(MergeCellRule.Grid.of(0, 4), MergeCellRule.Grid.of(1, 4)).
map(MergeCellRule.Grid.of(rowRenderData.length - 1, 0), MergeCellRule.Grid.of(rowRenderData.length - 1, 3)).
build();
TableRenderData table = Tables.of(rowRenderData).mergeRule(rule).create();
// 数据封装
tableData.setTable(table);
// 传入模板模板地址+信息数据
XWPFTemplate template = XWPFTemplate.compile(wordPath + modelName).render(tableData);
// 指定下载的文件名--设置响应头
response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("附件1-报价明细表.docx", "UTF-8"));
response.setContentType("application/vnd.ms-excel;charset=UTF-8");
response.setHeader("Pragma", "no-cache");
response.setHeader("Cache-Control", "no-cache");
response.setDateHeader("Expires", 0);
try {
OutputStream out = response.getOutputStream();
BufferedOutputStream bos = new BufferedOutputStream(out);
template.write(out);
bos.flush();
out.flush();
template.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
来源:https://blog.csdn.net/CSDN_SAVIOR/article/details/128295169


猜你喜欢
- 不论是float 还是double都是浮点数,而计算机是二进制的,浮点数会失去一定的精确度。Java在java.math包中提供的API类B
- delegate double ProcessDelegate(double param1, double param2); &n
- 开放端口安全组没开放端口是原罪!!!导致好多BUG费时费力。Hbase悄悄 * 的用了好多端口,比如被我抓到的42239,直接搜索报错药不对症
- 环境Linux版本:CentOS 6.5、Ubuntu 12.04.5 JDK版本:JDK 1.7目录方法一:手动解压JDK的压缩包,然后设
- 这篇文章主要介绍了Java开发工具IntelliJ IDEA安装图解,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习
- 目录一、常用属性、方法和事件1.常用属性2.常用方法3.常用事件二、应用实例1.实例描述2.属性设置3.事件处理4.实现代码总结一、常用属性
- 一、概述一共两个线程,一个线程生产产品,一个线程消费产品,使用同步代码块方法,同步两个线程。当产品没有时,通知生产者生产,生产者生产后,通知
- 一、Socket是什么Socket 的中文翻译过来就是“套接字”。套接字是什么,我们先来看看它的英文含义:插座。Socket 就像一个电话插
- 本文实例为大家分享了Unity3D实现批量下载图片功能的具体代码,供大家参考,具体内容如下下一篇文章试试用线程下载代码如下using Sys
- 概要应同学邀请,演示如何使用 PyQt5 内嵌浏览器浏览网页,并注入 Javascript 脚本实现自动化操作。下面测试的是一个廉价机票预订
- 前言不得不说,JSP 现在已经是一门十分老旧的技术了,学习编程时,不仅要学习优秀的前言技术,还要对基础有一定的把握,所以学习 JSP 时,我
- 内发光原理内发光原理简单概况是:采样周边像素alpha取平均值叠加效果。概括来说似乎好像特别简单,但需要一定的理解和消化。发光物体可以当做是
- 下面就为大家带来3种比较常见的压缩方式先给出一组数据原图:width:2976; height:2976原图实际:--->byte:2
- 本文实例讲述了C#实现将记事本中的代码编译成可执行文件的方法,运行环境为VS2012,具体方法如下:1、在记事本中编写如下代码:using
- @MapperScan包扫描的坑在使用通用mapper执行查询时,由于不太注意顺手就导了spring的包:import org.mybati
- SQLite 数据库简介SQLite 是一个轻量级数据库,它是D. Richard Hipp建立的公有领域项目,在2000年发布了第一个版本
- C#调用C++ dll string类型返回为了这个问题,百度了一堆不靠谱的资料,什么C#调用c++类型对应啥的,说用string ,Str
- 项目中遇到了下载文件文件名是中文而且还有空格如果不对连接进行处理下载就会报错要想解决这个问题只需对你的url 进行编码然后替换空格用编码表示
- 首先需要在 AndroidManifest.xml 文件中添加「获取模拟定位信息」权限。<uses-permission androi
- 废话不多说,直接上代码import java.text.SimpleDateFormat; import java.util.Date;/*