基于Freemarker和xml实现Java导出word

作者:经典鸡翅 时间:2022-07-11 23:15:12 

前言

最近做了一个调查问卷导出的功能,需求是将维护的题目,答案,导出成word,参考了几种方案之后,选择功能强大的freemarker+固定格式之后的wordxml实现导出功能。导出word的代码是可以直接复用的,于是在此贴出,并进行总结,方便大家拿走。

实现过程概览

先在word上,调整好自己想要的样子。然后存为xml文件。保存为freemarker模板,以ftl后缀结尾。将需要替换的变量使用freemarker的语法进行替换。最终将数据准备好,和模板进行渲染,生成文件并返回给浏览器流。

详细的实现过程准备好word的样式

我们新建一个word,我们应该使用Microsoft office,如果使用wps可能会造成样式有些不兼容。在新建的office中,设置好我们的表格样式。我们的调查问卷涉及到四种类型,单选,多选,填空,简答。我们做出四种类型的示例。

基于Freemarker和xml实现Java导出word

样式没有问题后,我们选择另存为word xml 2003版本。将会生成一个xml文件。

基于Freemarker和xml实现Java导出word

格式化xml,并用freemarker语法替换xml

我们可以先下载一个工具 firstobject xml editor,这个可以帮助我们查看xml,同时方便我们定位我们需要改的位置。
复制过去之后,按f8可以将其进行格式化,左侧是标签,右侧是内容,我们只需要关注w:body即可。

基于Freemarker和xml实现Java导出word

像右侧的调查问卷这个就是个标题,我们实际渲染的时候应该将其进行替换,比如我们的程序数据map中,有title属性,我们想要这里展示,我们就使用语法${title}即可。

基于Freemarker和xml实现Java导出word

freemarker的具体语法,可以参考freemarker的问题,在这里我给出几个简单的例子。

比如我们将所有的数据放置在dataList中,所以我们需要判断,dataList是不是空,是空,我们不应该进行下面的逻辑,不是空,我们应该先循环题目是必须的,答案是需要根据类型进行再次循环的。语法参考文档,这里不再赘述。

程序端引入freemarker


<dependency>
 <groupId>org.freemarker</groupId>
 <artifactId>freemarker</artifactId>
</dependency>

将我们的flt文件放在resources下的templates下。

后端代码实现

此代码可以复用,在此贴出


public class WordUtils {

private static Configuration configuration = null;
 private static final String templateFolder = WordUtils.class.getClassLoader().getResource("").getPath()+"/templates/word";
 static {
   configuration = new Configuration();
   configuration.setDefaultEncoding("utf-8");
   try {
     configuration.setDirectoryForTemplateLoading(new File(templateFolder));
   } catch (IOException e) {
     e.printStackTrace();
   }
 }

/**
  * @Description:导出word,传入request,response,map就是值,title是导出问卷名,ftl是你要使用的模板名
  */
 public static void exportWord(HttpServletRequest request, HttpServletResponse response, Map map, String title, String ftlFile) throws Exception {
   Template freemarkerTemplate = configuration.getTemplate(ftlFile);
   File file = null;
   InputStream fin = null;
   ServletOutputStream out = null;
   try {
     file = createDocFile(map,freemarkerTemplate);
     fin = new FileInputStream(file);
     String fileName = title + ".doc";
response.setCharacterEncoding("utf-8");
response.setContentType("application/msword");
response.setHeader("Content-Disposition", "attachment;filename="
      +fileName);
out = response.getOutputStream();
     byte[] buffer = new byte[512];
     int bytesToRead = -1;
     while((bytesToRead = fin.read(buffer)) != -1) {
       out.write(buffer, 0, bytesToRead);
     }
   }finally {
     if(fin != null) fin.close();
     if(out != null) out.close();
     if(file != null) file.delete();
   }
 }

/**
  * @Description:创建doc文件
  */
 private static File createDocFile(Map<?, ?> dataMap, Template template) {
   File file = new File("init.doc");
   try {
     Writer writer = new OutputStreamWriter(new FileOutputStream(file), "utf-8");
     template.process(dataMap, writer);
     writer.close();
   } catch (Exception e) {
     e.printStackTrace();
   }
   return file;
 }

}

有了工具类后,我们准备好我们的map数据。map里面的数据大家可以自行定义。然后调用utils中的导出方法即可。

WordUtils.exportWord(request, response, dataMap, "word", "demo.ftl");

来源:https://www.cnblogs.com/jichi/p/11921541.html

标签:Freemarker,xml,Java
0
投稿

猜你喜欢

  • spring boot 全局异常处理方法汇总

    2021-07-06 22:44:04
  • Android实现自定义带文字和图片Button的方法

    2021-06-20 17:13:50
  • C语言中的回调函数实例

    2021-09-27 08:37:12
  • Android音乐播放器制作 加入控制台(三)

    2022-04-15 05:09:30
  • SpringBoot读取自定义配置文件方式(properties,yaml)

    2021-06-30 23:46:07
  • IDEA MyBatis Plugins自动生成实体类和mapper.xml

    2021-07-24 01:26:37
  • Spring Cloud Alibaba 整合Nacos的详细使用教程

    2022-10-19 22:18:00
  • Android 蓝牙2.0的使用方法详解

    2022-10-19 17:21:21
  • swagger如何返回map字段注释

    2023-02-22 08:56:27
  • RxJava 触发流基本原理源码解析

    2023-06-24 06:02:57
  • Android中bindService基本使用方法概述

    2023-08-05 19:18:56
  • 关于@MapperScan包扫描的坑及解决

    2023-02-13 02:45:46
  • C++11中的可变参数模板/lambda表达式

    2023-06-10 18:20:14
  • C#正则过滤HTML标签并保留指定标签的方法

    2022-12-23 18:21:39
  • SpringBoot 配置文件总结

    2021-09-06 13:12:57
  • Android开发中WebView的简单使用小结

    2022-09-11 00:22:43
  • Netty分布式高性能工具类recycler的使用及创建

    2022-03-04 17:57:32
  • Spring Cloud Eureka(全面解析) 大白话

    2022-11-12 22:43:02
  • 利用Lambda表达式创建新线程案例

    2023-08-26 23:16:00
  • JAVA实现红包分发的示例代码

    2022-10-08 06:18:15
  • asp之家 软件编程 m.aspxhome.com