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
0
投稿
猜你喜欢
- Zookeeper和Eureka哪个更好?1、CAP理论一个分布式系统不可能同时很好的满足一致性,可用性和分区容错性这三个需求C:数据一致性
- Ireport的安装及使用一、 安装ireport1.点击安装包,可以一直点击下一步,安装完成。2.安装完成后,要在如下安装目录下添加ojd
- 本文介绍了Flutter 通过Clipper实现各种自定义形状的示例代码,分享给大家,具体如下:ClipOval 圆形裁剪ClipOval(
- 昨天写this用法总结的时候,突然产生了一个问题,请教别人之后,有了自己的一点认识。还是把它写下来,为大家更好的认识提供一点思路。1)有人写
- 背景:SpringMVC如何响应json格式的数据?技术实现方式1:在Controller使用@RestController注解方式2:在C
- 介绍大家都知道微信支付的回调链接要求不能跟参数,但又要接收返回的xml数据。我开始使用@RequestBody注解在参数上,希望能获取xml
- 1. Java中程序的逻辑控制语句1.1顺序结构顺序结构比较简单,按照代码书写的顺序一行一行执行1.2分支结构1.2.1 switch语句这
- 前言我们之前学的单链表,默认只能从链表的头部遍历到链表的尾部,在实际中应用太少见,太局限;而双向链表,对于该链表中的任意节点,既可以通过该节
- 一、迭代key&value第一种方式:迭代entrySet1.方法一/** * entrySet集合for-each循环(
- 1、声明一个测试对象import java.time.LocalDate;import java.util.List;import lomb
- 在c#中可以遍历指定驱动器或指定目录下嵌套目录中的所有文件或者任意深度的文件。通过遍历可以检索string形式的目录名和文件名,也可以检索
- 消息过滤RocketMQ分布式消息队列的消息过滤方式有别于其它MQ中间件,是在Consumer端订阅消息时再做消息过滤的。RocketMQ这
- 大家都知道NPOI组件可以在你本地没有安装office的情况下来 读取,创建excel文件。但是大家一般都是只默认读取一个excel文件的第
- java类型转换 Integer String Long Float Double Date1如何将字串 String 转换成整数 int?
- 目录1.下面有关JSP内置对象的描述,说法错误的是?2.Java语言与其它语言相比,独有的特点是( )3.下面哪个选项正确创建socket连
- 在Web应用系统开发中,文件上传和下载功能是非常常用的
- 情况一:配置文件,无法被导出或者生效修改前:修改后:究其原因,这是由于Maven的约定大于配置,导致我们写的配置文件,无法被导出或者生效的问
- 本文实例讲述了C#从命令行读取参数的方法。分享给大家供大家参考。具体如下:using System;class MainClass{ pub
- 分析:label标签控件是主线程创建的,不能直接从另一个线程访问.可以这样认为:不能跨线程直接访问控件;最简单的办法就是:using Sys
- public String[] split(String regex) 默认limit为0public String[] split(Str