freemarker模板生成的word文档优化

发布时间:2024年01月07日

前言

	通过freemarker模板生成的word文档实际上不是真正意义上的ms word标准文档,它仍然是xml文件,而docx实际上是一个压缩文件。所以使用以下方法对freemarker模板生成的word进行优化,保证生成的文件是真正意义上的ms word标准文档。

解决方案

  1. 使用压缩工具打开docx模板,取出document.xml,如下图:
    在这里插入图片描述
  2. 将用压缩工具打开后的docx文档里面的document.xml复制出来,并将document.xml后缀改为.ftl,然后进行参数预设。
  3. 将内容格式化后修改需要替换的内容为freemarker标签,对document.ftl进行参数预设,如下图:
    在这里插入图片描述
  4. 文件准备好后存放到某个目录下,docx文件为将要生成好的docx文件样式模版,必须要有的。如下图:
    在这里插入图片描述
  5. 程序代码
import freemarker.template.Configuration;
import freemarker.template.Template;
 
import java.io.*;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import java.util.zip.ZipOutputStream;
 
public class Main {
 
    public static void main(String[] args) {
        //模版文件存放目录
        String templatepath = "E:/test/";
        //模版样式文件名
        String docxname = "document.docx";
        //模版xml文件
        String xmlname = "document.xml";
         //替换freemarker标签后的临时xml文件
        String tmpxmlpath = "E:/test/temp.xml";
        //最终生成的docx文件
        String targetpath = "E:/test/final.docx";
        // 数据
        Map<String,Object> data = new HashMap();
        data.put("summary","这里是替换的文字");
        // 生成文档
        try {
            outputWord(templatepath, docxname, xmlname, tmpxmlpath, targetpath, data);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
 
    /**
     * @Description 根据参数生成docx合同文档
     * @author HubertToLee
     * @param templatepath 模板所在文件夹
     * @param docxname docx格式模板文件名(不带路径)
     * @param xmlname xml格式模板,有freemaker标记(不带路径)
     * @param tmpxmlpath 临时xml文件路径
     * @param targetPath 目标文件路径
     * @param param 待填充数据
     * @return
     * @throws Exception
     */
    private static boolean outputWord(String templatepath, String docxname, String xmlname,
                                    String tmpxmlpath, String targetPath, Map<String, Object> param) throws Exception {
         //-------------start-----------------
        Configuration cfg = new Configuration();
        cfg.setDirectoryForTemplateLoading(new File(templatepath));
        //以UTF-8的编码格式读入文件
        Template template = cfg.getTemplate(xmlname,"UTF-8");
        //以UTF-8的编码格式输出文件
        template.setOutputEncoding("UTF-8");
        Writer out = new FileWriter(new File(tmpxmlpath));
        // 数据放到模板xml里面,生成带数据的xml
        template.process(param, out);
        if (out != null) {
            out.close();
        }
         //-------------end-----------------
		//以上这块代码目的只为生替换freemarker标签后的xml文件
		//实际开发中,也可以用替换后的字符串生成xml文件来生成word
        File file = new File(tmpxmlpath);
        File docxFile = new File(templatepath + "/" + docxname);
        ZipFile zipFile = new ZipFile(docxFile);
        Enumeration<? extends ZipEntry> zipEntrys = zipFile.entries();
        ZipOutputStream zipout = new ZipOutputStream(new FileOutputStream(targetPath));
        int len = -1;
        byte[] buffer = new byte[1024];
        while (zipEntrys.hasMoreElements()) {
            ZipEntry next = zipEntrys.nextElement();
            InputStream is = zipFile.getInputStream(next);
            // 把输入流的文件传到输出流中 如果是word/document.xml由我们输入
            zipout.putNextEntry(new ZipEntry(next.toString()));
            if ("word/document.xml".equals(next.toString())) {
                InputStream in = new FileInputStream(file);
                while ((len = in.read(buffer)) != -1) {
                    zipout.write(buffer, 0, len);
                }
                in.close();
            } else {
                while ((len = is.read(buffer)) != -1) {
                    zipout.write(buffer, 0, len);
                }
                is.close();
            }
        }
        zipout.close();
        return true;
    }
}
  1. 会在指定目录下生成文件。如下图:
    在这里插入图片描述
文章来源:https://blog.csdn.net/mcband/article/details/135443810
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。