一,简单模板导出(不含图片, 不含表格循环)
1, 新建一个word文档, 输入如下类容:
2, 将该word文件另存为xml格式(注意是另存为,不是直接改扩展名)
3, 将xml文件的扩展名直接改为ftl
4, 用java代码完成导出(需要导入freemarker.jar)
@Test public void exportSimpleWord() throws Exception{ // 要填充的数据, 注意map的key要和word中${xxx}的xxx一致 MapdataMap = new HashMap (); dataMap.put("username", "张三"); dataMap.put("sex", "男"); //Configuration用于读取ftl文件 Configuration configuration = new Configuration(); configuration.setDefaultEncoding("utf-8"); /*以下是两种指定ftl文件所在目录路径的方式, 注意这两种方式都是 * 指定ftl文件所在目录的路径,而不是ftl文件的路径 */ //指定路径的第一种方式(根据某个类的相对路径指定) //configuration.setClassForTemplateLoading(this.getClass(),""); //指定路径的第二种方式,我的路径是C:/a.ftl configuration.setDirectoryForTemplateLoading(new File("C:/")); // 输出文档路径及名称 File outFile = new File("D:/test.doc"); //以utf-8的编码读取ftl文件 Template t = configuration.getTemplate("a.ftl","utf-8"); Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "utf-8"),10240); t.process(dataMap, out); out.close(); }
5, 这时在D盘下就生成了一个test.word, 打开可以看到${xxx}已被替换
二, word文件中导入图片
1, 新建一个word文档, 在要插入图片的地方随便插入一张图片
2, 将word另存为xml
3, 将xml扩展名改为ftl
4, 打开ftl文件, 搜索w:binData 或者 png可以快速定位图片的位置,图片 已经编码成0-Z的字符串了, 如下:
5, 将上述0-Z的字符串全部删掉,写上${imgStr}(变量名随便写)后保存
6, 导入图片的代码与上述代码是一样的, 也是创建一个Map, 将数据存到map中,只不过我们要把图片用代码进行编码,将其也编成0-Z的字符串:
MapdataMap = new HashMap (); dataMap.put("imgStr", getImageStr()); //....其余省略
这是对图片进行编码的代码:
public String getImageStr() { String imgFile = "d:/aa.png"; 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); }
注意: 该代码需要用到 sun.misc.BASE64Encoder 类,这个类就是JDK中的类,但在eclipse中默认是不访问的,需要设置一下,设置方式:
项目上右键-->Build Path-->Configure Build Path...
双击Access rules,点击add, 选择Accessible,下方输入**, OK , 这样就可以访问sun.misc.BASE64Encoder 类了
三, 导出循环的表格
1, 新建一个word文档, 插入如下表格:
2, 另存为xml, 将扩展名改为ftl
3, 搜索 w:tr 可以找到行的起点与结束点(注意第一对w:tr 是表头,应找第二对 w:tr), 如图:
4, 用<#list userList as user> #list>标签将第二对 w:tr 标签包围起来(userList是集合的key, user是集合中的每个元素, 类似
5, 解析该ftl文件
这是User类
public class User { private String a; private String b; private String c; //生成set和get方法,此处省略 }
这是解析ftl文件的代码,跟上面一样,只是Map的value是一个集合而已
@Test public void exportListWord() throws Exception{ //构造数据 MapdataMap = new HashMap (); List list = new ArrayList (); for(int i=0;i<10;i++){ User user = new User(); user.setA("a"+(i+1)); user.setB("b"+(i+1)); user.setC("c"+(i+1)); list.add(user); } dataMap.put("userList", list); Configuration configuration = new Configuration(); configuration.setDefaultEncoding("utf-8"); configuration.setDirectoryForTemplateLoading(new File("C:/")); File outFile = new File("D:/test.doc"); Template t = configuration.getTemplate("c.ftl","utf-8"); Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "utf-8"),10240); t.process(dataMap, out); out.close(); }
如果你需要输出集合的索引, 用${user_index}即可.
四, 常见问题解决方案
4.1, 异常信息如下:
freemarker.core.ParseException: Encountered "<" at line 3, column 28888 in test.ftl. Was expecting one of:... ... "false" ... "true" ... ... ... "." ... "+" ... "-" ... "!" ... "[" ... "(" ...
这是由于在写${xxx}表达式的时候, xxx与其前方的文字样式不一致, 在另存为xml后你可以搜索一下 "${" , 会发现如下图这种情况:
由于${xxx}中的xxx格式与其前方的文字不一致, 那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签, 所以在word中看似写的是${xxx}实际上转为xml后变成了${
五, javaWeb中利用response导出(注意编码问题,防止中文乱码)
MapdataMap = new HashMap (); dataMap.put("username", "张三"); dataMap.put("sex", "男"); Configuration configuration = new Configuration(); configuration.setDefaultEncoding("utf-8"); configuration.setDirectoryForTemplateLoading(new File(request.getRealPath("/")+"/templete"));//指定ftl所在目录,根据自己的改 response.setContentType("application/msword"); response.setHeader("Content-Disposition", "attachment;filename=\"" + new String("文件名.doc".getBytes("GBK"), "iso8859-1") + "\""); response.setCharacterEncoding("utf-8");//此句非常关键,不然word文档全是乱码 PrintWriter out = response.getWriter(); Template t = configuration.getTemplate("test.ftl","utf-8");//以utf-8的编码读取ftl文件 t.process(dataMap, out); out.close();