我的编程空间,编程开发者的网络收藏夹
学习永远不晚

SpringBoot动态导出word文档(完美实整教程 复制即可使用,不能实现你找我)

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

SpringBoot动态导出word文档(完美实整教程 复制即可使用,不能实现你找我)

背景

最近有一个需求是需要动态导出合同、订单等信息,导出一个word文档供客户进行下载查看。

需要导出的word文件,主要可以分为两种类型。

  1. 导出固定内容和图片的word文档
  2. 导出表格内容不固定的word文档

经过对比工具,我实践过两种实现方式。第一种是FreeMarker模板来进行填充;第二种就是文中介绍的POI-TL。

这里我推荐使用POI-TL

介绍

POI-TL是word模板引擎,基于Apache POI,提供更友好的API。

目前最新的版本是1.12.X,POI对应版本是5.2.2。

这里需要注意的是POI和POI-TL有一个对应的关系。

在这里插入图片描述

准备工作

我使用的POI-TL版本是1.10.0

在这里插入图片描述

            com.deepoove            poi-tl            1.10.0                            org.apache.poi            poi            4.1.2                            org.apache.poi            poi-ooxml            4.1.2                            org.apache.poi            poi-ooxml-schemas            4.1.2                            commons-io            commons-io            2.7        

快速开始

流程:制作模板->提供数据->渲染模板->下载word

注意:需要填充的数据需要使用{{}}来表示。

1. 导出固定内容和图片的word文档

准备模板

在这里插入图片描述

模板保存为docx格式,存放在resource目录下

在这里插入图片描述

提供数据

private Map assertMap() {        Map params = new HashMap<>();        params.put("name", "努力的蚂蚁");        params.put("age", "18");        params.put("image", Pictures.ofUrl("http://deepoove.com/images/icecream.png").size(100, 100).create());        return params;    }

工具方法

    private String copyTempFile(String templeFilePath) {        InputStream inputStream = getClass().getClassLoader().getResourceAsStream(templeFilePath);        String tempFileName = System.getProperty("user.home") + "/" + "1.docx";        File tempFile = new File(tempFileName);        try {            FileUtils.copyInputStreamToFile(inputStream, tempFile);        } catch (IOException e) {            throw new RuntimeException(e);        }        return tempFile.getPath();    }
private void down(HttpServletResponse response, String filePath, String realFileName) {        String percentEncodedFileName = null;        try {            percentEncodedFileName = percentEncode(realFileName);        } catch (UnsupportedEncodingException e) {            throw new RuntimeException(e);        }        StringBuilder contentDispositionValue = new StringBuilder();        contentDispositionValue.append("attachment; filename=").append(percentEncodedFileName).append(";").append("filename*=").append("utf-8''").append(percentEncodedFileName);        response.addHeader("Access-Control-Allow-Origin", "*");        response.addHeader("Access-Control-Expose-Headers", "Content-Disposition,download-filename");        response.setHeader("Content-disposition", contentDispositionValue.toString());        response.setHeader("download-filename", percentEncodedFileName);        try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream(filePath));             // 输出流             BufferedOutputStream bos = new BufferedOutputStream(response.getOutputStream());) {            byte[] buff = new byte[1024];            int len = 0;            while ((len = bis.read(buff)) > 0) {                bos.write(buff, 0, len);            }        } catch (Exception e) {            e.printStackTrace();        }    }
    public static String percentEncode(String s) throws UnsupportedEncodingException {        String encode = URLEncoder.encode(s, StandardCharsets.UTF_8.toString());        return encode.replaceAll("\\+", "%20");    }

编写接口

@RequestMapping("genera")    public void genera(HttpServletResponse response) {        //1.组装数据        Map params = assertMap();        //2.获取根目录,创建模板文件        String path = copyTempFile("word/1.docx");        String fileName = System.currentTimeMillis() + ".docx";        String tmpPath = "D:\\" + fileName;        try {            //3.将模板文件写入到根目录            //4.编译模板,渲染数据            XWPFTemplate template = XWPFTemplate.compile(path).render(params);            //5.写入到指定目录位置            FileOutputStream fos = new FileOutputStream(tmpPath);            template.write(fos);            fos.flush();            fos.close();            template.close();            //6.提供前端下载            down(response, tmpPath, fileName);        } catch (Exception e) {            e.printStackTrace();        } finally {            //7.删除临时文件            File file = new File(tmpPath);            file.delete();            File copyFile = new File(path);            copyFile.delete();        }    }

对于图片的格式,POI-TL也提供了几种方式来提供支撑。

在这里插入图片描述

测试

请求接口:http://127.0.0.1:1000/file/genera

效果如下:

在这里插入图片描述

2. 导出表格内容不固定的word文档

表格动态内容填充,POI-TL提供了3种方式。

  1. 表格行循环
  2. 表格列循环
  3. 动态表格。

第二种和第三种都可以实现表格填充,但我个人感觉第一种更方便一点,这里我只介绍【表格行循环】实现方式。

LoopRowTableRenderPolicy 是一个特定场景的插件,根据集合数据循环表格行。

注意:

  1. 模板中有两个list,这两个list需要置于循环行的上一行。
  2. 循环行设置要循环的标签和内容,注意此时的标签应该使用[]

准备模板

在这里插入图片描述

提供数据

学生实体类

public class Student {    private String name;    private String age;    public String getName() {        return name;    }    public void setName(String name) {        this.name = name;    }    public String getAge() {        return age;    }    public void setAge(String age) {        this.age = age;    }}

学生word类

public class StudentTable {    private String title;    private List<Student> studentList;    private List<Student> studentList1;    public List<Student> getStudentList1() {        return studentList1;    }    public void setStudentList1(List<Student> studentList1) {        this.studentList1 = studentList1;    }    public String getTitle() {        return title;    }    public void setTitle(String title) {        this.title = title;    }    public List<Student> getStudentList() {        return studentList;    }    public void setStudentList(List<Student> studentList) {        this.studentList = studentList;    }}

表格数据

private StudentTable assertData() {        StudentTable table = new StudentTable();        table.setTitle("我是标题");        List<Student> studentList = new ArrayList<>();        Student student = new Student();        student.setName("张三");        student.setAge("18");        studentList.add(student);        Student student1 = new Student();        student1.setName("李四");        student1.setAge("20");        studentList.add(student1);        Student student2 = new Student();        student2.setName("王五");        student2.setAge("21");        studentList.add(student2);        Student student3 = new Student();        student3.setName("马六");        student3.setAge("19");        studentList.add(student3);        table.setStudentList(studentList);        table.setStudentList1(studentList);        return table;    }

编写接口

@RequestMapping("dynamicTable")    public void dynamicTable(HttpServletResponse response) {        //1.组装数据        StudentTable table = assertData();        //2.获取根目录,创建模板文件        String path = copyTempFile("word/2.docx");        //3.获取临时文件        String fileName = System.currentTimeMillis() + ".docx";        String tmpPath = "D:\\" + fileName;        try {            //4.编译模板,渲染数据            LoopRowTableRenderPolicy hackLoopTableRenderPolicy = new LoopRowTableRenderPolicy();            Configure config =                    Configure.builder().bind("studentList", hackLoopTableRenderPolicy).bind("studentList1", hackLoopTableRenderPolicy).build();            XWPFTemplate template = XWPFTemplate.compile(path, config).render(table);            //5.写入到指定目录位置            FileOutputStream fos = new FileOutputStream(tmpPath);            template.write(fos);            fos.flush();            fos.close();            template.close();            //6.提供下载            down(response, tmpPath, fileName);        } catch (Exception e) {            e.printStackTrace();        } finally {            //7.删除临时文件            File file = new File(tmpPath);            file.delete();            File copyFile = new File(path);            copyFile.delete();        }    }

测试

请求接口:http://127.0.0.1:1000/file/dynamicTable

效果如下:
在这里插入图片描述

来源地址:https://blog.csdn.net/qq_37758497/article/details/128858175

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

SpringBoot动态导出word文档(完美实整教程 复制即可使用,不能实现你找我)

下载Word文档到电脑,方便收藏和打印~

下载Word文档

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录