Java生成word文档(freemarker)

来源:互联网 发布:苏州软件测试培训班 编辑:程序博客网 时间:2024/06/05 18:59

近期工作需要,需将一些计算结果导出为excel、word格式的文档,结果为一个稍复杂的二维表格。参考网上诸多前辈的博客,现将方法记录如下,参考链接会放至文末。

首先是生成word文档,需要有个模板,例:



上图:是Word文档中的内容,也就是模板,为了下面步鄹做铺垫,所以在需要输入数据的地方改成了拼音,


将word文档另存为xml文件.




接下来,上面写的拼音就起到作用了.

打开xml文件.搜索 title.

将Title 改为 ${title}

其他地方一样的修改。

改完后,把文件的后缀名直接改成ftl。


然后呢,直接上代码:

[java] view plain copy
  1. package com;  
  2.   
  3. import java.io.BufferedWriter;  
  4. import java.io.File;  
  5. import java.io.FileNotFoundException;  
  6. import java.io.FileOutputStream;  
  7. import java.io.IOException;  
  8. import java.io.OutputStreamWriter;  
  9. import java.io.Writer;  
  10. import java.util.ArrayList;  
  11. import java.util.HashMap;  
  12. import java.util.List;  
  13. import java.util.Map;  
  14.   
  15. import freemarker.template.Configuration;  
  16. import freemarker.template.Template;  
  17. import freemarker.template.TemplateException;  
  18.   
  19. public class WordTest {  
  20.       
  21.     private Configuration configuration = null;  
  22.       
  23.     public WordTest(){  
  24.         configuration = new Configuration();  
  25.         configuration.setDefaultEncoding("UTF-8");  
  26.     }  
  27.       
  28.     public static void main(String[] args) {  
  29.         WordTest test = new WordTest();  
  30.         test.createWord();  
  31.     }  
  32.       
  33.     public void createWord(){  
  34.         Map<String,Object> dataMap=new HashMap<String,Object>();  
  35.         getData(dataMap);  
  36.         configuration.setClassForTemplateLoading(this.getClass(), "/com");  //FTL文件所存在的位置  
  37.         Template t=null;  
  38.         try {  
  39.             t = configuration.getTemplate("wordModel.ftl"); //文件名  
  40.         } catch (IOException e) {  
  41.             e.printStackTrace();  
  42.         }  
  43.         File outFile = new File("O:/outFilessa"+Math.random()*10000+".doc");  
  44.         Writer out = null;  
  45.         try {  
  46.             out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile)));  
  47.         } catch (FileNotFoundException e1) {  
  48.             e1.printStackTrace();  
  49.         }  
  50.            
  51.         try {  
  52.             t.process(dataMap, out);  
  53.         } catch (TemplateException e) {  
  54.             e.printStackTrace();  
  55.         } catch (IOException e) {  
  56.             e.printStackTrace();  
  57.         }  
  58.     }  
  59.   
  60.     private void getData(Map<String, Object> dataMap) {  
  61.         dataMap.put("title""标题");  
  62.         dataMap.put("year""2012");  
  63.         dataMap.put("month""2");  
  64.         dataMap.put("day""13");  
  65.         dataMap.put("auditor""唐鑫");  
  66.         dataMap.put("phone""13020265912");  
  67.         dataMap.put("weave""占文涛");  
  68. //      dataMap.put("number", 1);  
  69. //      dataMap.put("content", "内容"+2);  
  70.           
  71.         List<Map<String,Object>> list = new ArrayList<Map<String,Object>>();  
  72.         for (int i = 0; i < 10; i++) {  
  73.             Map<String,Object> map = new HashMap<String,Object>();  
  74.             map.put("number", i);  
  75.             map.put("content""内容"+i);  
  76.             list.add(map);  
  77.         }  
  78.           
  79.           
  80.         dataMap.put("list", list);  
  81.     }  
  82. }  

大家现在可能最关心的是遍历的文件。接下来我们看如何遍历数据。

freemarker的用法:http://www.havenliu.com/goodarticle/420.html

在刚刚的那个ftl文件中直接编辑:
打开FTL文件,搜索   <w:tr>
找到第一个,tr的意思不用解释了吧,代表着一行。
这也意味着我找到了Table中的第一行,但是我需要遍历的不是从第一行开始,而是从第二行。
好的,继续搜索,找到第二个。



在它的头上加一个<#list 你的集合名称 as xxxx>
ok,有开头就有结尾,玩过JSP上的EL表达式应该不会感觉很陌生吧。
搜索 </w:tr>
同样找到第二个,加上</#list>结束符。
好的,这样就没问题了。

如上所示,如果加上<#list list as test>的话,须得在list包含的那一行占位符里,将${XXX}改为${test.XXX},这样才能将值传进去。

tips:

//Configuration用于读取ftl文件        Configuration configuration = new Configuration();        configuration.setDefaultEncoding("utf-8");                /*以下是两种指定ftl文件所在目录路径的方式, 注意这两种方式都是        * 指定ftl文件所在目录的路径,而不是ftl文件的路径        */        //指定路径的第一种方式(根据某个类的相对路径指定)        //configuration.setClassForTemplateLoading(this.getClass(),"");                //指定路径的第二种方式,我的路径是C:/a.ftl        configuration.setDirectoryForTemplateLoading(new File("C:/"));  

总结来讲就是:

1.建Word模板,其中用占位符${XXX}来表示要填充的项,如果有需循环的表格,可设为${test.XXX}(test和下面某处呼应);

2.将文档另存为.xml格式,然后用sublime等编辑器打开,搜索${,看看${XXX}是否被拆开了,如果被拆开了,删掉多余项,没有的话保存为.ftl文件(直接改后缀即可);

3.如果模板中有表格,需要动态循环增加的项,看一下占位符所在行,如上图例子所示为第二行,则在ftl文件中搜<w:tr>,找到第二个,在前面添加<#list 你的集合名字 as test>(你的集合名字,是在代码中放到map里的key,而as后的值也可以自定义,本文取为test,其要与ftl文件中的<#list>标签所包含的占位符所一直,这些占位符表示为${test.XXX},要对应),接着搜索</w:tr>,找到第二个,在其后加上</#list>,保存文件。

4.将ftl文件放置某目录下,然后参考上面写代码,注意map的key要和占位符名对应,不过list里的可以不加test。

5.还有导入freemarker.jar包。


更多可参考:

1.Java生成Word文档

2.java导出word之freemarker导出

3.Java Web项目中使用Freemarker生成Word文档

4.http://www.360doc.com/content/13/0731/10/13247663_303740756.shtml

5.Java 动态生成复杂 Word

excel导出包含合并单元格可参考:

6.Java poi导出excel合并标题单元格