java导出生成word文档_java使用freemarker 生成word文档
最近需要做一个导出word的功能, 在网上搜了下, 有用POI,JXL,iText等jar生成一个word文件然后将数据写到该文件中,API非常繁琐而且拼出来的样式也不美观,于是选择了另一种方式----feemarker基于word模板的导出方式, 这种方式非常简单而且导出的样式美观, 其原理就是先做一个word模板, 该模板中变量数据用${xxx}这种方式填写, 然后再导出时只需读取模板然后用相应的数据替换其中的${xxx}即可.
一,简单模板导出(不含图片, 不含表格循环)
1, 新建一个word文档, 输入如下类容:
2, 将该word文件另存为xml格式(注意是另存为,不是直接改扩展名)
3, 将xml文件的扩展名直接改为ftl
4, 用java代码完成导出(需要导入freemarker.jar)
package test;
import java.io.BufferedWriter;
import java.io.File;
import java.io.FileOutputStream;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.util.HashMap;
import java.util.Map;
import freemarker.template.Configuration;
import freemarker.template.Template;
public class Test {
public static void main(String[] args) throws Exception {
testExportSimpleWord();
}
public static void testExportSimpleWord() throws Exception {
// 要填充的数据, 注意map的key要和word中${xxx}的xxx一致
Map dataMap = new HashMap();
dataMap.put("whb", "汪海波");
// 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("C:/test.doc");
// 以utf-8的编码读取ftl文件
Template t = configuration.getTemplate("test.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的字符串:
Map dataMap = 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>标签将第二对 w:tr 标签包围起来(userList是集合的key, user是集合中的每个元素, 类似), 如图:
5, 解析该ftl文件
这是User类
Java代码
publicclassUser {
privateString a;
privateString b;
privateString c;
//生成set和get方法,此处省略
}
这是解析ftl文件的代码,跟上面一样,只是Map的value是一个集合而已
public void exportListWord() throws Exception{
//构造数据
Map dataMap = 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, 异常信息如下:
Xml代码
freemarker.core.ParseException: Encountered "
Was expecting one of:
...
...
"false" ...
"true" ...
...
...
"." ...
"+" ...
"-" ...
"!" ...
"[" ...
"(" ...
这是由于在写${xxx}表达式的时候, xxx与其前方的文字样式不一致, 在另存为xml后你可以搜索一下 "${" , 会发现如下图这种情况:
由于${xxx}中的xxx格式与其前方的文字不一致, 那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签, 所以在word中看似写的是${xxx}实际上转为xml后变成了${xxx},这样这个el表达式中的标签就解析不了报错了, 可以去掉${}内部的标签只留下xxx或者删掉 "${" 和 "}"然后给xxx加上el表达式都可以解决此问题.
五, javaWeb中利用response导出(注意编码问题,防止中文乱码)
Map dataMap = 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();
java导出生成word文档_java使用freemarker 生成word文档相关推荐
- java导出pdf文件并下载_java根据模板生成pdf文件并导出
1.首先需要依赖包:itext的jar包,我是maven项目,所以附上maven依赖 [html] view plain copy com.itextpdf itextpdf 5.5.10 [html ...
- java导出excel 打不开_Java使用POI生成Excel文件后打不开的问题
在实际的工作中,有时会遇到获取数据后需要存入Excel文件的情况.但是,在生成Excel文件后,发现无法正常打开该文件. 例如:以当前的时间点为文件名,新生成一个Excel文件.先来看看下面一段代码. ...
- java 操作word中表格_Java 使用Spire.Cloud.Word给Word文档添加表格
在编辑Word文档时,很多时候需要用到表格,以便能够清晰整洁地表达和归类数据.本文就将介绍如何使用Spire.Cloud.Word给Word文档添加表格.Spire.Cloud.Word提供了Tabl ...
- 生成html页面的ftl文件,FreeMarker生成静态HTML页面的工具类FreeMarkerUtil
FreeMarker生成静态HTML页面的工具类FreeMarkerUtil 一.FreeMarkerUtil工具类: import com.huaxia.entity.News; import co ...
- java循环导出word文档_Java使用freemarker导出word文档
通过freemarker,以及JAVA,导出word文档. 共分为三步: 第一步:创建模板文件 第二步:通过JAVA创建返回值. 第三步:执行 分别介绍如下: 第一步: 首先创建word文档,按照想要 ...
- java类似word校验错字_Java 比较两个Word文档差异
Java 比较两个Word文档差异 Java 比较两个Word文档差异 本文介绍使用Spire.Doc for Java的比较功能来比较两个相似Word文档的差异.需要使用的版本为3.8.8或者后续发 ...
- java生成word 框勾_Java 使用模板生成 Word 文件---基于 Freemarker 模板框架
Java项目引入 Freemarker 插件自行完成. 步骤如下: 1.编写 Word 模板,并将模板中要用代码动态生成数据用 Freemarker 变量取代,即${变量名},如${username} ...
- java生成word样式变形的解决方案(freemarker生成word的样式不对)
word的样式不对,被挤走了,怎么调dtl文件 文章目录 前言 一.现在的样式问题 1.现象1_(样式被挤掉) 2.现象2_(多了一个空白页,也是被挤下去的) 3.生成的word是代码 4.样式不对, ...
- java word 题目导入_java使用poi导入word题库
java使用poi导入word题库 包含单选,多选,填空,判断 //word图示 //具体代码,仅适合本人自己的格式 //代码一 InputStream inputStream = file.getI ...
最新文章
- 速卖通运营之商品结构及分层逻辑
- 小米手机安装https证书报错:无法安装该证书 因为无法读取该证书文件
- Coding:从给定数字集中找到最大的数字
- Galera Cluster for MySQL 集群恢复
- spring boot微服务通用部署启动脚本
- Heartbeat+DRBD+MySQL高可用架构方案与实施过程细节
- 开课吧Java课堂:字符串如何处理?
- Learn the python the hard way (Day 3)
- java bean jsonobject_利用JSONObject将json 字符串转换为java bean对象
- 计算机Excel应用案例,Excel VBA在Office中的应用案例
- bootstrap-入门学习-流体容器与响应式布局容器
- 河南大学大计算机考研复试分数线预测,2017年河南大学考研复试分数线以及复试通知...
- 动态主题模型(Dynamic Topic Models, DTM)
- 我们为什么要推广经方?
- 计算机上没有端口DOT4,dot4_001端口
- 参数校验@Valid
- 网易2017校园招聘数据挖掘笔试题
- 课题组王猛的论文被遥感领域顶级期刊 IEEE TGRS 录用
- Javascript 计算请假天数
- 服务器里那个文件是地图的爆率,关于普通图爆率研究