源地址:http://blog.csdn.net/y_love_f/article/details/41596375?utm_source=tuicool&utm_medium=referral
上一篇博客中讲FreeMarker做了简单的介绍,这篇博客就来使用它来完成两个例子。
在项目中使用它完成的功能是按照固定的模板将数据导出到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包放到程序中,然后将上面制作好的模板放到程序中;
最后加上下面的代码就可以了
- package com.ftl;
- import java.io.*;
- 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;
- import sun.misc.BASE64Encoder;
- public class DocumentHandler {
- private Configuration configuration = null;
- public DocumentHandler() {
- configuration = new Configuration();
- configuration.setDefaultEncoding("utf-8");
- }
- public void createDoc() {
- // 要填入模本的数据文件
- Map dataMap = new HashMap();
- getData(dataMap);
- // getTest(dataMap);
- // 设置模本装置方法和路径,FreeMarker支持多种模板装载方法。可以重servlet,classpath,数据库装载,
- // 这里我们的模板是放在com.template包下面
- configuration.setClassForTemplateLoading(this.getClass(),
- "/com/template");
- Template t = null;
- try {
- // test.ftl为要装载的模板
- t = configuration.getTemplate("temple.ftl");
- t.setEncoding("utf-8");
- } catch (IOException e) {
- e.printStackTrace();
- }
- // 输出文档路径及名称
- File outFile = new File("D:/test.doc");
- Writer out = null;
- try {
- out = new BufferedWriter(new OutputStreamWriter(
- new FileOutputStream(outFile), "utf-8"));
- } catch (Exception e1) {
- e1.printStackTrace();
- }
- try {
- t.process(dataMap, out);
- out.close();
- } catch (TemplateException e) {
- e.printStackTrace();
- } catch (IOException e) {
- e.printStackTrace();
- }
- }
- /**
- * 注意dataMap里存放的数据Key值要与模板中的参数相对应
- * @param dataMap
- *
- */
- @SuppressWarnings("unchecked")
- private void getData(Map dataMap) {
- dataMap.put("image", getImageStr());
- dataMap.put("name", "张三");
- List<Map<String, Object>> newsList=new ArrayList<Map<String,Object>>();
- for(int i=1;i<=5;i++){
- Map<String, Object> map=new HashMap<String, Object>();
- map.put("purchaseTime", "进货日期"+i);
- map.put("product", "产品名称"+i);
- map.put("factory", "生产厂家"+i);
- map.put("spec", "产品规格"+i));
- map.put("number", "进货数量"+i);
- newsList.add(map);
- }
- dataMap.put("newsList",newsList);
- }
- private String getImageStr() {
- String imgFile = "d:/1.png";
- InputStream in = null;
- byte[] data = null;
- try {
- in = new FileInputStream(imgFile);
- data = new byte[in.available()];
- in.read(data);
- in.close();
- } catch (IOException e) {
- e.printStackTrace();
- }
- BASE64Encoder encoder = new BASE64Encoder();
- return encoder.encode(data);
- }
- }
- package com.ftl;
- public class main {
- /**
- * @param args
- */
- public static void main(String[] args) {
- DocumentHandler dh=new DocumentHandler();
- dh.createDoc();
- System.out.println("end");
- }
- }
3. 生成的结果展示
4. 总结
这个工具生成word确实很方便,但是通过这两天查找资料也发现这个工具也是有很多不足的地方的,毕竟不是什么都是十全十美的。在网上找到一个帖子简单的总结了一下这个工具的三个不好的地方,下面是连接地址有确定的可以去看看。http://www.iteye.com/topic/17468