freemarker生成word模版文档及list便利数据

freemarker生成word模版文档及list便利数据(二)

      在项目中使用它完成的功能是按照固定的模板将数据导出到Word。比如台账。在完成后将处理过程按照台账的要求导出,有时程序中需要实现生成标准Word文档,要求能够打印,并且保持页面样式不变。

     这个功能就是由XML+Freemarder来实现的,Word从2003开始支持XML格式,大致的步骤:用office2003或者以上的版本编辑好 word的样式,然后另存为xml,将xml翻译为FreeMarker模板,最后用java来解析FreeMarker模板并输出Doc。

      上一篇博客中提到,使用Freemarker其实就只准备模板和数据。下面的步骤其实概括起来就两步:第一步是制作模板,第二步是准备数据.

具体实现过程如 下:

1、  制作模板

     首先用office【版本要2003以上,以下的不支持xml格式】编辑文档的样式,将需要动态填充的内容使用Freemarker标签替换:Word文档样式如下:

        

 

将Word文档另存为XML格式,将后缀名“xml”修改为“ftl”,模板制作完成

图片处理:

    我们可以看到上面的模板中有图片,当我们将xml修改为ftl之后打开文件可以看到原来图片的位置全部都是乱码。这是因为ftl不能够识别图片,我们只要将这些乱码用${image}替换掉既可以了。

        

循环处理:

模板中明细表中的数据是一个集合,我们需要循环遍历才能够将数据全部注入,这个时候我们就会用到了FreeMarker本身的语法了。

循环:

<#list array as bean>

  ${bean.property}

</#list>

     代码放在表格的开头和结尾,也即是将表格中的标签包括在循环里面,这样子就可以了。

在制作模板的过程中遇到的一个问题:${xxx}这些标记在ftl中被拆成一个一个的,让我很无语。最后我还是手动的将这些又重新拼到一起了。

2. 准备数据,将上面模板中的${}的标签放上对应的数据

首先要讲freemarker的jar包放到程序中,然后将上面制作好的模板放到程序中;

           

 

最后加上下面的代码就可以了

 

[java] 

  1. package com.ftl;  
  2.   
  3. import java.io.*;  
  4. import java.util.ArrayList;  
  5. import java.util.HashMap;  
  6. import java.util.List;  
  7. import java.util.Map;  
  8. import freemarker.template.Configuration;  
  9. import freemarker.template.Template;  
  10. import freemarker.template.TemplateException;  
  11. import sun.misc.BASE64Encoder;  
  12.   
  13. public class DocumentHandler {  
  14.   
  15.     private Configuration configuration = null;  
  16.   
  17.     public DocumentHandler() {  
  18.         configuration = new Configuration();  
  19.         configuration.setDefaultEncoding("utf-8");  
  20.     }  
  21.   
  22.     public void createDoc() {  
  23.          // 要填入模本的数据文件  
  24.         Map dataMap = new HashMap();  
  25.         getData(dataMap);  
  26. //      getTest(dataMap);  
  27.         // 设置模本装置方法和路径,FreeMarker支持多种模板装载方法。可以重servlet,classpath,数据库装载,    
  28.           // 这里我们的模板是放在com.template包下面    
  29.         configuration.setClassForTemplateLoading(this.getClass(),  
  30.         "/com/template");  
  31.         Template t = null;  
  32.         try {  
  33.              // test.ftl为要装载的模板   
  34.             t = configuration.getTemplate("temple.ftl");  
  35.             t.setEncoding("utf-8");  
  36.   
  37.         } catch (IOException e) {  
  38.             e.printStackTrace();  
  39.         }  
  40.         // 输出文档路径及名称  
  41.         File outFile = new File("D:/test.doc");  
  42.         Writer out = null;  
  43.           
  44.         try {  
  45.             out = new BufferedWriter(new OutputStreamWriter(  
  46.             new FileOutputStream(outFile), "utf-8"));  
  47.         } catch (Exception e1) {  
  48.             e1.printStackTrace();  
  49.         }  
  50.   
  51.         try {  
  52.             t.process(dataMap, out);  
  53.             out.close();  
  54.         } catch (TemplateException e) {  
  55.             e.printStackTrace();  
  56.         } catch (IOException e) {  
  57.             e.printStackTrace();  
  58.         }  
  59.     }  
  60.       
  61.     /** 
  62.      * 注意dataMap里存放的数据Key值要与模板中的参数相对应  
  63.      * @param dataMap 
  64.      *  
  65.      */  
  66.     @SuppressWarnings("unchecked")  
  67.     private void getData(Map dataMap) {  
  68.          dataMap.put("image", getImageStr());  
  69.          dataMap.put("name", "张三");    
  70.            
  71.          List<Map<String, Object>> newsList=new ArrayList<Map<String,Object>>();  
  72.             for(int i=1;i<=5;i++){  
  73.              Map<String, Object> map=new HashMap<String, Object>();  
  74.              map.put("purchaseTime", "进货日期"+i);  
  75.              map.put("product", "产品名称"+i);  
  76.              map.put("factory", "生产厂家"+i);  
  77.              map.put("spec", "产品规格"+i));  
  78.              map.put("number", "进货数量"+i);  
  79.              newsList.add(map);  
  80.             }  
  81.             dataMap.put("newsList",newsList);  
  82.     }  
  83.      private String getImageStr() {  
  84.          String imgFile = "d:/1.png";  
  85.          InputStream in = null;  
  86.          byte[] data = null;  
  87.          try {  
  88.              in = new FileInputStream(imgFile);  
  89.              data = new byte[in.available()];  
  90.              in.read(data);  
  91.              in.close();  
  92.          } catch (IOException e) {  
  93.              e.printStackTrace();  
  94.          }  
  95.          BASE64Encoder encoder = new BASE64Encoder();  
  96.          return encoder.encode(data);  
  97.      }  
  98.   
  99. }  


[java] 

  1. package com.ftl;  
  2.   
  3. public class main {  
  4.     /** 
  5.      * @param args 
  6.      */  
  7.     public static void main(String[] args) {  
  8.         DocumentHandler dh=new DocumentHandler();  
  9.         dh.createDoc();  
  10.         System.out.println("end");  
  11.     }  
  12.   
  13. }  

 

3. 生成的结果展示

 

4. 总结

       这个工具生成word确实很方便,但是通过这两天查找资料也发现这个工具也是有很多不足的地方的,毕竟不是什么都是十全十美的。

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在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.
以下是一个使用 FreeMarker 生成 Word 文档的示例。 首先需要在 pom.xml 中添加以下依赖: ```xml <dependency> <groupId>org.freemarker</groupId> <artifactId>freemarker</artifactId> <version>2.3.30</version> </dependency> <dependency> <groupId>org.apache.poi</groupId> <artifactId>poi-ooxml</artifactId> <version>4.1.2</version> </dependency> ``` 然后可以创建一个 FreeMarker 模板文件,例如 `template.ftl`: ```html <html> <head> <style> table { border-collapse: collapse; width: 100%; } th, td { border: 1px solid black; text-align: left; padding: 8px; } th { background-color: #ddd; } </style> </head> <body> <h1>${title}</h1> <table> <tr> <th>Name</th> <th>Age</th> <th>Email</th> </tr> <#list users as user> <tr> <td>${user.name}</td> <td>${user.age}</td> <td>${user.email}</td> </tr> </#list> </table> </body> </html> ``` 这个模板包含了一个表格,其中列出了用户的姓名、年龄和电子邮件地址。 然后可以使用以下代码读取模板并将数据填充到模板中: ```java import freemarker.template.Configuration; import freemarker.template.Template; import org.apache.poi.xwpf.usermodel.*; import java.io.*; import java.util.*; public class WordGenerator { public static void generate(Map<String, Object> data, OutputStream outputStream) throws Exception { // 创建 FreeMarker 配置 Configuration cfg = new Configuration(Configuration.VERSION_2_3_30); cfg.setClassForTemplateLoading(WordGenerator.class, "/"); cfg.setDefaultEncoding("UTF-8"); // 加载模板 Template template = cfg.getTemplate("template.ftl"); // 创建 Word 文档 XWPFDocument document = new XWPFDocument(); // 将模板填充到 Word 文档中 StringWriter writer = new StringWriter(); template.process(data, writer); String html = writer.toString(); InputStream inputStream = new ByteArrayInputStream(html.getBytes()); XWPFHtmlConverter.getInstance().convert(inputStream, document); // 保存 Word 文档 document.write(outputStream); outputStream.close(); } public static void main(String[] args) throws Exception { // 准备数据 Map<String, Object> data = new HashMap<>(); data.put("title", "User List"); List<Map<String, Object>> users = new ArrayList<>(); users.add(createUser("Alice", 25, "alice@example.com")); users.add(createUser("Bob", 30, "bob@example.com")); users.add(createUser("Charlie", 35, "charlie@example.com")); data.put("users", users); // 生成 Word 文档 generate(data, new FileOutputStream("output.docx")); } private static Map<String, Object> createUser(String name, int age, String email) { Map<String, Object> user = new HashMap<>(); user.put("name", name); user.put("age", age); user.put("email", email); return user; } } ``` 这个示例将生成一个名为 `output.docx` 的 Word 文档,其中包含了填充后的表格。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值