Home Java中使用FreeMarker导出Word文档
Post
Cancel

Java中使用FreeMarker导出Word文档

制作模板

在制作模板时需要注意的一些问题如下:

  1. 用Microsoft Office Word 2003或以上的版本来制作,不要用WPS做。
  2. 关于图片,一定要插入一张图片,方便后面找到图片的位置好进行替换。
  3. 一些需要动态生成的数据,如个人信息,用personInfo(其他字段也行,建议和程序中的VO字段对应)来代替,注意,单词要先用一个文本框写好,再复制黏贴过来,不要在Word上手动敲,不然再转成xml时单词会被拆散,也不要直接黏贴过来一个${personInfo},还是会被拆散,最好直接黏贴一个单词过来,在xml中再找到这个单词再添加${}
  4. 表格只需要画一行,循环在xml中配置。

模板画好之后,直接将Word另存为xml格式,保存完后将Word关掉,用Notepad++打开保存的xml格式的文档找到你写的单词如stdName。

在单词前后加上${},所有你写的单词都加上。

搜索w:tr找到表格。

在表格标签<w:tr>的首尾分别加上<#list expList as exp></#list>标签,并且标签里的单词要改成${exp.time}这种形式。解释一下:list标签可以使标签内的表格自动循环生成相应的行数,expList是自己命名的,是表格的数据来源,是一个list集合,exp是别名。

搜索<pkg:binaryData>,你会发现这个标签里的内容特别多,这就是你画模板时插入的图片的BASE64字符串。

<pkg:binaryData>标签里的内容删完,替换为${images},这样就可以在代码中设置,然后导出自己获取的图片了,当然名字自己取,叫img也可以,只要在代码中和这对应就好了。

将上面修改好的xml文档保存好,点击重命名将其类型修改为ftl。至此,模板就制作好了,在后面代码中要用的模板就是template.ftl。

Java代码

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
private void exportWord(String ssdw, String image64Str, String stdName, String personInfo, List<FamilyInfo> familyInfos,
                        Map<String, String> maps) {
    Map<String, Object> dataMap = new HashMap<String, Object>();
    // ssdw 是职员信息
    dataMap.put("orgName", ssdw);
    // image64Str 是职员的照片转换后的BASE64字符串
    dataMap.put("images", image64Str);
    // stdName 是职员信息
    dataMap.put("stdName", stdName);
    // personInfo 是职员信息
    dataMap.put("personInfo", personInfo);

    // familyList 是职员的家庭信息的集合,对应模板中的表格
    List<Map<String, Object>> familyList = new ArrayList<Map<String, Object>>();
    for (int i = 0; i < familyInfos.size(); i++) {
        Map<String, Object> map = new HashMap<String, Object>();
        map.put("gx", familyInfos.get(i).getYbrgx());
        map.put("xm", familyInfos.get(i).getCyxm());
        map.put("dwjzw", familyInfos.get(i).getDwjzw());
        familyList.add(map);
    }
    dataMap.put("familyList", familyList);

    // expList 是职员的工作经历信息的集合,对应模板中的表格
    List<Map<String, Object>> expList = new ArrayList<Map<String, Object>>();
    Iterator<Entry<String, String>> it = maps.entrySet().iterator();
    while (it.hasNext()) {
        Map<String, Object> map = new HashMap<String, Object>();
        Map.Entry<String, String> entity = (Entry<String, String>) it.next();
        map.put("time", entity.getKey());
        map.put("experience", entity.getValue());
        expList.add(map);
    }
    dataMap.put("expList", expList);

    @SuppressWarnings("deprecation")
    Configuration configuration = new Configuration();
    configuration.setDefaultEncoding("utf-8");
    // 有两种方式获取你的模板,模板在项目中时用第一个,模板在本地时用第二个。
    // 注意:两种方式的路径都只需要写到模板的上一级目录
    configuration.setClassForTemplateLoading(this.getClass(), "/tem");
    // configuration.setDirectoryForTemplateLoading(new File("C:/"));  

    File outFile = new File("D:/outFilessa" + Math.random() * 10000 + ".doc"); // 输出路径
    Template t = null;  
    Writer out = null;
    try {
        t = configuration.getTemplate("template.ftl", "utf-8"); // 获取模板
        out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "utf-8")); 

        // 如果想由浏览器指定下载位置,则使用下面代码
        // String filename = "模板.doc";
        // response.setContentType("application/ms-excel;charset=UTF-8");
        // response.setHeader("Content-Disposition", "attachment;filename=".concat(String.valueOf(URLEncoder.encode(filename, "UTF-8"))));
        // out = new BufferedWriter(new OutputStreamWriter(response.getOutputStream(), "utf-8"));

        t.process(dataMap, out);
    } catch (Exception e) {
        e.printStackTrace();  
    } finally {
        try {
            out.close();
        } catch (IOException  e1) {
            e.printStackTrace(); 
        }
    }
}
This post is licensed under CC BY 4.0 by the author.