Java+freemaker+xml生成word模板

date: 2017-04-15 21:34:56


Java利用freemaker包来操作生成word模板。


一个头疼的问题,了解了下xml。
Java也是在学习的路上。


需求

自动化生成word报告,需要现有的word模板,选择后自动填入所需参数和计算后的结果。
可以减少人必要的输入,提高效率,提高准确率。

R1:静态文字word模板

Step1

该方法需要先手动创建一个doc模板,并保存为xml文件。
通过动态替换特定标签${}中的内容生成。

word形式:

Step2

通过word制作好模板,另存为xml文件。
里面的${}会被分开,需要删除多余的东西。只需要留下${}及{}里面的标识符。

例如这样:

Step3

处理好xml文件后就可以写Java程序,需要注意是的标识符一致。
Java程序结构:


Java程序:
DocUtil.java

package creatreport;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.util.Map;

import freemarker.template.Configuration;
import freemarker.template.Template;
import freemarker.template.TemplateException;
import freemarker.template.TemplateExceptionHandler;
import sun.misc.BASE64Encoder;

public class DocUtil {
    public Configuration configure=null;

    public DocUtil(){
//       configure=new Configuration(Configuration.VERSION_2_3_22);
         configure=new Configuration();
         configure.setDefaultEncoding("utf-8");
    }
    /**
     * 根据Doc模板生成word文件
     * @param dataMap 需要填入模板的数据
     * @param downloadType 文件名称
     * @param savePath 保存路径
     */
     public void createDoc(Map<String,Object> dataMap,String downloadType,String savePath){

        try {
            //加载需要装填的模板
            Template template=null;

            //设置模板装置方法和路径,FreeMarker支持多种模板装载方法。可以从servlet,classpath,数据库装载。
            //加载模板文件,放在testDoc下
            configure.setClassForTemplateLoading(this.getClass(), "/testDoc");

            //设置对象包装器
            //configure.setObjectWrapper(new DefaultObjectWrapper());

            //设置异常处理器
            configure.setTemplateExceptionHandler(TemplateExceptionHandler.IGNORE_HANDLER);

            //定义Template对象,注意模板类型名字与downloadType要一致
            template=configure.getTemplate(downloadType + ".xml");

            File outFile=new File(savePath);
            Writer out=null;
            //指定编码表需使用转换流,转换流对象要接收一个字节输出流
            out=new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "utf-8"));
            template.process(dataMap, out);
            out.close();
        } catch (IOException e) {
            e.printStackTrace();
        } catch (TemplateException e) {
            e.printStackTrace();
        }


    }

     public String getImageStr(String imgFile){
         InputStream in=null;
         byte[] data=null;
         try {
             in=new FileInputStream(imgFile);
             data=new byte[in.available()];
             in.read(data);
             in.close();
         } catch (FileNotFoundException e) {
             e.printStackTrace();
         } catch (IOException e) {
             e.printStackTrace();
         }
         BASE64Encoder encoder=new BASE64Encoder();
         return encoder.encode(data);
     }
}


TestDoc.java

package creatreport;

import java.util.*;

public class TestDoc {
    public static void main(String[] args){
        DocUtil docUtil=new DocUtil();
        Map<String, Object> dataMap=new HashMap<String, Object>();


        /**表4-1-1*/

        //weights
        dataMap.put("w1", "0.7");
        dataMap.put("w2", "0.18");
        dataMap.put("w3", "0.12");
        dataMap.put("w4", "0.02");
        dataMap.put("w5", "0.01");
        dataMap.put("w6", "0.3");
        dataMap.put("w7", "0.3");
        dataMap.put("w8", "0.28");
        dataMap.put("w9", "0.7");
        dataMap.put("w10", "0.02");
        dataMap.put("w11", "0.4");
        dataMap.put("w12", "0.25");
        dataMap.put("w13", "0.1");
        dataMap.put("w14", "0.1");
        dataMap.put("w15", "0.1");
        dataMap.put("w16", "0.05");
      //PS
        dataMap.put("ps1", "/");
        dataMap.put("ps2", "/");
        dataMap.put("ps3", "/");
        dataMap.put("ps4", "/");
        dataMap.put("ps5", "/");
        dataMap.put("ps6", "/");
        dataMap.put("ps7", "/");
        dataMap.put("ps8", "/");
        dataMap.put("ps9", "/");
        dataMap.put("ps10", "/");
        dataMap.put("ps11", "/");
        dataMap.put("ps12", "/");
        dataMap.put("ps13", "/");
        dataMap.put("ps14", "/");
        dataMap.put("ps15", "/");
        dataMap.put("ps16", "/");
      docUtil.createDoc(dataMap,"4_1_1","D:\\eclipseWorkspace\\bridgereport\\4_1_1.doc");
        System.out.println("Word文件已生成完毕!目录地址:D:\\eclipseWorkspace\\bridgereport\\4_1_1.doc");
    }
}

需要注意的问题:

  • 加入jar包后需要 build path。
  • xml模板放在testDoc下。

Step4

效果:

R2:静态图文模板

在模板里插入图片的情况。
建立word模板的时候,需要在之后插入图片的地方先任意插入一张图片占位。word另存为xml时候,${}该删除的和以前一样。但是在插入图片的地方会有一大堆编码。处理的方法是删除这堆编码,在图片的标志位下换成自己定义的标识符:


该留下的东西如上。
插入图片的代码:

    //将图片转换成BASE64字符串
    public String getImageStr(String imgFile){
        InputStream in=null;
        byte[] data=null;
        try {
            in=new FileInputStream(imgFile);
            data=new byte[in.available()];
            in.read(data);
            in.close();
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
     BASE64Encoder encoder=new BASE64Encoder();
     return encoder.encode(data);
    }

TestDoc中的使用代码:

        dataMap.put("leftPic1", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic1.png"));
        dataMap.put("leftPic2", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic2.png"));
        dataMap.put("rightPic1", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic3.png"));
        dataMap.put("rightPic2", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic4.png"));

需要主要的问题:

  • 图片的那堆编码删除后的形式如上图。
  • BASE64Decoder类不属于JDK标准库范畴,需要这样做,不然会报错:


图文静态效果:


R3:模板中有循环列表

需求:

本意是需要构件编号-备注是可以动态增加的,而原桥左幅和原桥右幅都只是一个单元格。
但是现在只能实现每行所有的列的动态增加。

xml修改的核心代码:

    <#list rightBanList as rightBan>

        ${rightBan.idR}

        ${rightBan.sortR}

        ${rightBan.detailR}
    </#list>

java循环类:

package liangban_report;

public class LeftBan {
    private String idL;
    private String sortL;
    private String detailL;
    private String rankL;
    private double scoreL;
    private String psL;
    public String getIdL() {
        return idL;
    }
    public void setIdL(String idL) {
        this.idL = idL;
    }
    public String getSortL() {
        return sortL;
    }
    public void setSortL(String sortL) {
        this.sortL = sortL;
    }
    public String getDetailL() {
        return detailL;
    }
    public void setDetailL(String detailL) {
        this.detailL = detailL;
    }
    public String getRankL() {
        return rankL;
    }
    public void setRankL(String rankL) {
        this.rankL = rankL;
    }
    public double getScoreL() {
        return scoreL;
    }
    public void setScoreL(double scoreL) {
        this.scoreL = scoreL;
    }
    public String getPsL() {
        return psL;
    }
    public void setPsL(String psL) {
        this.psL = psL;
    }

}

**TestDoc: **


package liangban_report;
import java.util.*;
public class TestDoc {
    public static void main(String[] args){
       //左幅右幅
        DocUtil docUtil=new DocUtil();
        Map<String, Object> dataMap=new HashMap<String, Object>();
        List<LeftBan> leftBanList = new ArrayList<LeftBan>();
        List<RightBan> rightBanList = new ArrayList<RightBan>();

        LeftBan leftBan1 = new LeftBan();
        leftBan1.setIdL("原左-1-1#板");
        leftBan1.setSortL("裂缝");
        leftBan1.setDetailL("板底出现9条横向裂缝,长度/宽度为30~50cm/0.05~0.1mm");
        leftBan1.setRankL("2");
        leftBan1.setScoreL(65.0);
        leftBan1.setPsL("/");
        leftBanList.add(leftBan1);

        LeftBan leftBan2 = new LeftBan();
        leftBan2.setIdL("原左-2-6#板");
        leftBan2.setSortL("渗水");
        leftBan2.setDetailL("板底出现渗水泛白,S=120X200cm2");
        leftBan2.setRankL("2");
        leftBan2.setScoreL(75.0);
        leftBan2.setPsL("/");
        leftBanList.add(leftBan2);

        dataMap.put("leftBanList", leftBanList);

        RightBan rightBan1 = new RightBan();
        rightBan1.setIdR("原右-1-8#板");
        rightBan1.setSortR("裂缝"+"  "+"露筋");
        rightBan1.setDetailR("板底出现6条横向裂缝,长度/宽度为80~100cm/0.05mm,4处单根露筋L=40cm,1处区域露筋,S=20X20cm2");
        rightBan1.setRankR("2"+"   "+"2");
        rightBan1.setScoreR(62.4);
        rightBan1.setPsR("/");
        rightBanList.add(rightBan1);

        RightBan rightBan2 = new RightBan();
        rightBan2.setIdR("原右-3-7#板");
        rightBan2.setSortR("破损");
        rightBan2.setDetailR("板底出现1处破损,S=120X150cm2");
        rightBan2.setRankR("3");
        rightBan2.setScoreR(60.0);
        rightBan2.setPsR("/");
        rightBanList.add(rightBan2);

        dataMap.put("rightBanList", rightBanList);

        dataMap.put("leftPic1", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic1.png"));
        dataMap.put("leftPic2", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic2.png"));
        dataMap.put("rightPic1", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic3.png"));
        dataMap.put("rightPic2", docUtil.getImageStr("D:\\eclipseWorkspace\\bridge_report_v2_0\\pic4.png"));

        docUtil.createDoc(dataMap,"liangban","D:\\eclipseWorkspace\\bridge_report_v2_0\\bridge_report_v2_0.doc");
        System.out.println("Doc文件已生成成功!");




    }
}

效果

问题:需要的效果没能做出来。

R4:以换行的形式来显示动态的word模板


在处理显示多行的xml中,并加换行符:

<#list firstDeductItem as firstItem>
  <w:t>${firstItem}</w:t><w:br/>
</#list>

TestDoc.java中改为:

        /*显示构件数量,如需增加数据,使用Str.add("")即可*/
        List<String> Strs=new ArrayList<String>();
        Strs.add("100");
        Strs.add("200");
        Strs.add("300");
        dataMap.put("brulcNum", Strs);

DocUtil.java中改为:

//定义Template对象,注意模板类型名字与downloadType要一致
template=configure.getTemplate(downloadType+".ftl");

此时xml文件会报错,当然也不能编译运行项目,需要将.xml文件改为.ftl文件保存。再编译运行,效果图:


以换行的形式来表示增加的单元格。
优化的方法再想想吧。
这个周末累死了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 199,711评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 83,932评论 2 376
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 146,770评论 0 330
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,799评论 1 271
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,697评论 5 359
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,069评论 1 276
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,535评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,200评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,353评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,290评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,331评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,020评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,610评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,694评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,927评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,330评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,904评论 2 341

推荐阅读更多精彩内容