freemarker实现word文档模板动态生成

携手创作,共同成长!这是我参与「掘金日新计划 · 8 月更文挑战」的第29天,点击查看活动详情

1.写在前面

很多时候,我们可能需要根据一个word模板,动态生成,我们所需要得一个word文档。

那这里,通过word模板生成word文档,我之前写过一个案例: [word模板数据填充并转pdf]

这个案例得word模板,有什么弊端呢?我们来看一下这个word模板,格式如下:

这样得一个word模板,我之前得做法是 替换段落文本 ,那会有啥问题?

  1. 文本的格式可能会丢失。
  2. 替换内容标识,可能会读取不到(单词别隔开)。
  3. 无法动态循环输出一个list集合。
  4. table表格无法做到动态输出。

当然啦,因为我们之前的业务功能比较简单,使用之前的 替换段落文本 方式,也是能实现到,所以就一直没有去研究,有无更好的方式。

嘿,随着业务功能的不断深入,动态list集合,动态table,这些功能也要求要实现了。

那我们作为一个程序员,这不得深入研究嘛?

巧了,哥们这几天,也研究了一个方式,可以实现动态list集合,动态table的渲染,那就是 freemarker模板引擎

好了,废话不多说,直接进入主题了!!!

2.freemarker实现

首先说一下实现步骤:通过将word模板,另存为 .xml 格式,然后将 .xml 文件后缀改成 .ftl ,然后再使用freemarker模板引擎,将数据填充到 .ftl ,然后再输出成一个 word 文档。

实现步骤,还是挺清晰的。那接下来,我们来实现一下:

2.1 word模板

例如:我们有一个这样的api接口文档,然后我们的系统后台会配置管理了很多api接口,然后导出到一个word文件。

1.输出的word文件,要带上文档目录结构(左边)

2.api的内容,要有table表格

3.api接口是一个list集合

由此可见,如果还是用我以前的方式,估计实现起来就相当麻烦了。

那这里用freemarker,如何实现呢?freemarker的语法是:${xxxx},那我们来改造模板,格式如下

这里为什么要用:${api.xxx}呢?我们来看一下freemarker是如何渲染一个list的?

<#list apiList as api> 
  ${api.apiName}   
</#list>

看到这,估计大家都懂了把?

就是说我们有一个 apiList 集合对象, apiList as api ,别名为 api ,进行遍历。 ${api.apiName} 就输出了apiName对应的属性的值。

看到这,我们的上面的模板,就很好理解了把?那我们来另存为xml。

2.2 .xml改.ftl

使用idea打开.ftl,然后格式化一下,搜索一下 ${api.apiName}

这里要注意一下,可能会有些地方报错,例如下面:

对于这些内容,我们改一下即可。

参考 ${api.apiName} 的格式即可。

好了,这里会有个问题,我们并没有看到有渲染一个list集合的?那这里,我们只能自己去构造了。

找到所有需要动态list渲染的内容,然后再加上 <#list > 标签

看到这里,估计大家都知道怎么改 .ftl 模板了吧?这里就不在一一列举了。

2.3 freemarker渲染

  • 如果是使用springboot,那我们的pom.xml文件,可以添加下面的依赖:
<!-- freemarker-starter -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-freemarker</artifactId>
    <version>2.3.12.RELEASE</version>
</dependency>

当然,直接引入fremarker依赖也是可以的,例如下面:

<dependency>
  <groupId>org.freemarker</groupId>
  <artifactId>freemarker</artifactId>
  <version>2.3.31</version>
</dependency>
  • 代码实现
@Autowired
private Configuration configuration;

@GetMapping("/doExportApiDoc2Word")
public void doExportApiDoc2Word(HttpServletResponse response,
                                @ApiParam @RequestParam(value = "projectId") String projectId){
    try {
        //返回word文档
        String fileName = URLEncoder.encode("接口文档" + DateUtil.format(new Date(), "yyyyMMddHHmmss"), "UTF-8");
        response.setCharacterEncoding("UTF-8");
        response.addHeader("Content-Disposition", "attachment; filename="" + fileName + ".docx"");

        //获取apiDoc所需要的数据
        Map<String, Object> dataModel = llsydnService.getApiDocData(projectId);

        //加载模板
        Template template = configuration.getTemplate("apiDoc.ftl", "utf-8");
        //渲染模板
        template.process(dataModel, response.getWriter());
        //response的Writer不需要我们手动关,tomcat会帮我们关的
    } catch (Exception e) {
        log.error("导出word异常:", e);
    }
}

由上可见,基本上都不需要我们操作,我们只需要构造出对象的数据,传给freemarker引擎即可。

我们要做的,就是定义好 .ftl 模板,构造好所需的参数。

这里说明一下,这个 apiDoc.ftl 模板,我们默认是放在templates目录下面即可,例如下面:

2.4 导出word

好了,我们测试一下,浏览器直接访问: http://localhost:8080/doExportApiDoc2Word?projectId=1

由此可见,导出的效果,还是蛮好的。

嘿,功能也实现了,我真是个聪明的小伙!!!_

今天,又是充实的一天!!!

哎呀,一股风吹过,头顶不禁有点凉意。

啥情况? 我的头发呢?(卑微)

好了,以上就是 freemarker实现word文档模板动态生成 的分享了。

个人实操可能也不够全面,班门弄斧了。

如果觉得有收获的,帮忙 点赞、评论、收藏 一下呗!!!

  • 3
    点赞
  • 10
    收藏
  • 打赏
    打赏
  • 3
    评论
在Web应用中,有时需要按照固定的模板将数据导出到Word,如流程审批单,在流程处理完成后将处理过程按照流程单的要求导出,有时程序中需要实现生成标准Word文档,要求能够打印,并且保持页面样式不变,常见的方案有POI、iText、JACOB、JSP几种方式,POI读取Word文档比较适合、对于生成文档样式比较难控制,iText操作Excel还可以,对Word的操作功能有限,JACOB操作Word实现复杂,并且无法将服务部署到Linux平台,要求安装office,对于实现固定格式的报表实现困难,对于JSP直接输出方式样式控制难。 Word从2003开始支持XML格式,用XML+Freemarder还做就很简单了,大致的思路是先用office2003或者2007编辑好 word的样式,然后另存为xml,将xml翻译为FreeMarker模板,最后用java来解析FreeMarker模板并输出Doc。经测试这样方式生成word文档完全符合office标准,样式、内容控制非常便利,打印也不会变形,生成的文档和office中编辑文档完全一样。具体实现过程如下: 1、 首先用office【版本要2003以上,以下的不支持xml格式】编辑文档的样式,将需要动态填充的内容使用Freemarker标签替换:Word文档样式如下: 2、 将Word文档另存为XML格式,将后缀名“xml”修改为“ftl” 3、 使用Freemarker填充内容,代码如下: [java] view plaincopyprint? 1. package com.test.freemarker.report; 2. 3. 4. 5. import java.io.BufferedWriter; 6. 7. import java.io.File; 8. 9. import java.io.FileOutputStream; 10. 11. import java.io.IOException; 12. 13. import java.io.OutputStreamWriter; 14. 15. import java.io.Writer; 16. 17. import java.util.HashMap; 18. 19. import java.util.Map; 20. 21. 22. 23. import freemarker.template.Configuration; 24. 25. import freemarker.template.Template; 26. 27. import freemarker.template.TemplateException; 28. 29. 30. 31. public class DocumentHandler { 32. 33. private Configuration configuration = null; 34. 35. 36. 37. public DocumentHandler() { 38. 39. configuration = new Configuration(); 40. 41. configuration.setDefaultEncoding("utf-8"); 42. 43. } 44. 45. 46. 47. public void createDoc() { 48. 49. // 要填入模本的数据文件 50. 51. Map dataMap = new HashMap(); 52. 53. getData(dataMap); 54. 55. // 设置模本装置方法和路径,FreeMarker支持多种模板装载方法。可以重servlet,classpath,数据库装载, 56. 57. // 这里我们的模板是放在com.havenliu.document.template包下面 58. 59.

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

敲代码的老贾

你的鼓励将是我创作的最大动力

¥2 ¥4 ¥6 ¥10 ¥20
输入1-500的整数
余额支付 (余额:-- )
扫码支付
扫码支付:¥2
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值