近期工作需要,需将一些计算结果导出为excel、word格式的文档,结果为一个稍复杂的二维表格。参考网上诸多前辈的博客,现将方法记录如下,参考链接会放至文末。
首先是生成word文档,需要有个模板,例:
上图:是Word文档中的内容,也就是模板,为了下面步鄹做铺垫,所以在需要输入数据的地方改成了拼音,
将word文档另存为xml文件.
接下来,上面写的拼音就起到作用了.
打开xml文件.搜索 title.
将Title 改为 ${title}
其他地方一样的修改。
改完后,把文件的后缀名直接改成ftl。
然后呢,直接上代码:
package com; import java.io.BufferedWriter; import java.io.File; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.OutputStreamWriter; import java.io.Writer; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import java.util.Map; import freemarker.template.Configuration; import freemarker.template.Template; import freemarker.template.TemplateException; public class WordTest { private Configuration configuration = null ; public WordTest(){ configuration = new Configuration(); configuration.setDefaultEncoding("UTF-8" ); } public static void main(String[] args) { WordTest test = new WordTest(); test.createWord(); } public void createWord(){ Map<String,Object> dataMap=new HashMap<String,Object>(); getData(dataMap); configuration.setClassForTemplateLoading(this .getClass(), "/com" ); Template t=null ; try { t = configuration.getTemplate("wordModel.ftl" ); } catch (IOException e) { e.printStackTrace(); } File outFile = new File( "O:/outFilessa" +Math.random()* 10000 + ".doc" ); Writer out = null ; try { out = new BufferedWriter( new OutputStreamWriter( new FileOutputStream(outFile))); } catch (FileNotFoundException e1) { e1.printStackTrace(); } try { t.process(dataMap, out); } catch (TemplateException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } } private void getData(Map<String, Object> dataMap) { dataMap.put("title" , "标题" ); dataMap.put("year" , "2012" ); dataMap.put("month" , "2" ); dataMap.put("day" , "13" ); dataMap.put("auditor" , "唐鑫" ); dataMap.put("phone" , "13020265912" ); dataMap.put("weave" , "占文涛" ); List<Map<String,Object>> list = new ArrayList<Map<String,Object>>(); for ( int i = 0 ; i < 10 ; i++) { Map<String,Object> map = new HashMap<String,Object>(); map.put("number" , i); map.put("content" , "内容" +i); list.add(map); } dataMap.put("list" , list); } }
大家现在可能最关心的是遍历的文件。接下来我们看如何遍历数据。
在刚刚的那个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合并标题单元格