最近一直在做导出功能,公司的页面太复杂,每个word页面的字段都有上百个,还要写近十个页面的导出,表示真心伤不起。技术也由刚开始的磕磕碰碰,到现在的熟练。

之前也曾做过导出功能(导出试卷),当时采用的是poi,直接输出试题,页面比较简单。这一次因为需要导出的word除了基本的文字信息外,还包含图片,复选框,表格循环显示等。选择了使用freemarker进行导出,百度到的都是先写好word,然后另存为xml,再将扩展名更改为ftl。然后通过Java代码填充数据,最终输出word。但遇到的问题是,我使用word-->xml-->ftl,得到的ftl模板在有的位置无法识别freemarker的一些标记,在无意之中发现通过word-->html--->ftl,也可以成功导出Word。

  1. 先写好所需要导出的word的模板,调整word的样式。将word另存为html页面,再更改扩展名为ftl。

  2. 编写后台的代码,部分代码如下。

public String createWord(){    Map
 dataMap = new HashMap
();    // word中需要展示的动态数据,用map集合来保存    dataMap.put("test", test);    WordUtil.createWord(dataMap, "test.ftl", filePath, fileOnlyName);    return "createWordSuccess";}public String dowloadWord() {                /** 先判断文件是否已生成  */        try {         //解决中文乱码         filePath = URLDecoder.decode(filePath, "UTF-8");         fileOnlyName = URLDecoder.decode(fileOnlyName, "UTF-8");         fileName = URLDecoder.decode(fileName, "UTF-8");                  //如果文件不存在,则会跳入异常,然后可以进行异常处理            new FileInputStream(filePath + File.separator +  fileOnlyName);        } catch (Exception e) {         e.printStackTrace();         return "error";        }        return "dowloadWord";    }    public InputStream getWordFile(){                try {                        //解决中文乱码            filePath = URLDecoder.decode(filePath, "UTF-8");            fileOnlyName = URLDecoder.decode(fileOnlyName, "UTF-8");            fileName = URLDecoder.decode(fileName, "UTF-8");            /** 返回最终生成的word文件流  */            return new FileInputStream(filePath + File.separator + fileOnlyName);        } catch (Exception e) {            e.printStackTrace();            return null;        }    }

WordUtil工具类代码实现如下:

import java.io.BufferedWriter;import java.io.File;import java.io.FileInputStream;import java.io.FileOutputStream;import java.io.InputStream;import java.io.OutputStreamWriter;import java.io.Writer;import java.util.Map;import sun.misc.BASE64Encoder;import freemarker.template.Configuration;import freemarker.template.Template;public class WordUtil {            public static String getImageStr(String imgFile) {        //String imgFile = "d:/aa.jpg";        InputStream in = null;        byte[] data = null;        try {            in = new FileInputStream(imgFile);            data = new byte[in.available()];            in.read(data);            in.close();        } catch (Exception e) {            e.printStackTrace();        }        BASE64Encoder encoder = new BASE64Encoder();        return encoder.encode(data);    }        @SuppressWarnings("unchecked")    public static void createWord(Map dataMap,String templateName,String filePath,String fileName){        try {                        //创建配置实例             Configuration configuration = new Configuration();                       //设置编码            configuration.setDefaultEncoding("UTF-8");                   //ftl模板文件统一放至com.fbty.coast.template 包下面            configuration.setClassForTemplateLoading(WordUtil.class,"/com/template/");                       //获取模板             Template template = configuration.getTemplate(templateName);                        //输出文件            File outFile = new File(filePath+File.separator+fileName);                        //如果输出目标文件夹不存在,则创建            if (!outFile.getParentFile().exists()){                outFile.getParentFile().mkdirs();            }                        //将模板和数据模型合并生成文件             Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile),"UTF-8"));            //生成文件            template.process(dataMap, out);                                    //关闭流            out.flush();            out.close();            } catch (Exception e) {                e.printStackTrace();            }    }}

对于填充的文本数据,在ftl模板中使用${text}获取,或者${text?default('')}。

实现复选框选中:

<#list ['强 ','中','弱'] as crack_dzpzx_list>        
<#if crack_dzpzx?exists>           <#list crack_dzpzx as crack_dzpzx_x>                               <#if crack_dzpzx_x==crack_dzpzx_list> checked="checked" 
          
                 
> ${crack_dzpzx_list?default('')}