Java中使用freemarker导出word文档(详细案例,含图片和表格)

Java中使用freemarker导出word文档(自己真实项目案例)

最近做了一个项目,有一个需求是要将职员的一些数据按照指定的样式导出成一个word文档出来,由于自己是一个Java小白,不懂,于是便在网上找方法,网上介绍的有很多,因为原来用poi导出过ppt,于是便想用poi来做,但是由于时间不够,最终选择了用freemarker来做,在做的过程中遇到了一些问题,但最终还是完成了。于是把自己完成这个功能所做的一切从头到尾的记录一编,希望能对其他人有帮助。


1.项目需要导出图片和表格

可以看出在这个样式中,需要导出图片和循环表格。

2.开始制作模板

在制作模板时,用office word 2003或以上的版本来制作,不要用wps做。在画模板遇到一些问题,我的解决如下:
1.关于图片,我画了一个文本框来放图片,然后随便放了一个图片进去(一定要插入一张图片,方便后面找到图片的位置好 进行替换)。
2.一些需要动态生成的数据写如个人信息用personInfo来代替,注意,单词要先用一个文本框写好,再复制黏贴过来,不要再word上手动敲,不然再转成xml时单词会被拆散。也不要直接黏贴过来一个${personInfo},我发现还是会被拆散,最好直接黏贴一个单词过来,在xml中再找到这个单词再添加${}。
3.表格只需要画一行,循环在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也可以,只要在代码中和这对应就好了。

3.完成模板

将上面修改好的xml文档保存好,点击重命名将其类型修改为ftl。我的是将template.xml改为template.ftl。
好了,至此,模板就制作好了,在后面代码中你要用的模板就是template.ftl。我是直接将template.ftl文档复制到我的项目中,当然,你放本地也可以,只是引用时方法和路径不一样而已。

4.Java代码

创建项目之类的就不说了,然后给你的项目到导freemarker的包。freemarker.jar包下载:
https://download.csdn.net/download/weixin_42142057/10651828
下面便是核心代码了:

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"));  
        t.process(dataMap, out);
    } catch (Exception e) {
        e.printStackTrace();  
    } finally {
        try {
            out.close();
        } catch (IOException  e1) {
            e.printStackTrace(); 
        }
    }

}

将获取的图片转换为BASE64字符串

public String getImageStr() {  
       //data是获取图片的二进制码;怎么获取图片以及怎么将图片转为二进制码的方法就不多说了
       BASE64Encoder encoder = new BASE64Encoder();  
       return encoder.encode(data);  
}  

要使用BASE64Encoder还得有如下操作:
点击你的项目名右键,选择Properties,进去后选择Java Build Path,然后如图:



操作完成,应用并OK。

Over 使用freemarker导出word完成。

SpringBoot_Freemarker生成Word_多个表格+两层嵌套循环; 步骤说明: 1.用Microsoft Office Word打开word原件;将文档需要动态生成的内容,替换为属性名 ${name} 2.另存为,选择保存类型Word 2003 XML 文档(*.xml) 3.用Firstobject free XML editor打开文件,选择Tools下的Indent【或者按快捷键F8】格式化文件内容。左边是文档结构,右边是文档内容; 4. 文档生成后有时需要手动修改,查找第一步设置的属性名,可能会产生类似${n.....ame}类似的样子,我们将将名字间的标签删掉,恢复为${name} 5. word模板表格,需要循环的位置, 用 标签将第二对 标签(即除表头的w:tr标签后的一对)包围起来 同时表格内的属性例如${name},在这里需要修改为${user.name} (userList是集合在dataMap的key, user是集合的每个元素, 类似), 如图: PLUS:若表格之外还有嵌套的循环,也需要用,注意这里的标签不要和某对其他标签交叉,不可以出现这种 6. 标识替换完之后,另存为.ftl后缀文件即可。 代码里是相对有一丢丢复杂的,两层嵌套循环; 总(dataMap) deptName 部门名 list(Table)表的集合 table1(map) table-名字 ${map.table} tableName-文名 ${map.tableName} columnCount-字段数 ${map.columnCount} recordCount-记录数 ${map.recordCount} listA-List--表格1 map.listA column Model属性——字段名 ${model.column} columnName Model属性——字段文名 ${model.column} rate Model属性——字段占比 ${model.rate} nullValueCount Model属性——字段空值数 ${model.nullValueCount} listB-List--表格2 map.listB …… listC-List--表格3 map.listC …… table2 table-名字 ${map.table} tableName-文名 ${map.tableName} columnCount-字段数 ${map.columnCount} recordCount-记录数 ${map.recordCount} listA-List--表格1 map.listA column Model属性——字段名 ${model.column} columnName Model属性——字段文名 ${model.column} rate Model属性——字段占比 ${model.rate} nullValueCount Model属性——字段空值数 ${model.nullValueCount} listB-List--表格2 map.listB …… listC-List--表格3 map.listC …… table3 ……
评论 12
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值