美文网首页我爱编程
Java+freemaker+xml生成word模板

Java+freemaker+xml生成word模板

作者: 朴有天虹 | 来源:发表于2017-06-06 15:49 被阅读0次

    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文件保存。再编译运行,效果图:


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

    相关文章

      网友评论

        本文标题:Java+freemaker+xml生成word模板

        本文链接:https://www.haomeiwen.com/subject/qmzgfxtx.html