JAVA 根据模板文件生成WORD文档
标签: java export word template
原文的原文链接:https://www.freesion.com/article/16261606/
原文链接:https://blog.csdn.net/zhangzhangjava/article/details/52937159

${xxx}不能带任何样式,否则不能将其内容进行替换,替换后的文字也是不带样式的,若要有样式可以直接编辑原始的xml文件增加样式

最近需要做一个导出word的功能, 在网上搜了下, 有用POI,JXL,iText等jar生成一个word文件然后将数据写到该文件中,API非常繁琐而且拼出来的样式也不美观,于是选择了另一种方式----feemarker基于word模板的导出方式, 这种方式非常简单而且导出的样式美观, 其原理就是先做一个word模板, 该模板中变量数据用xxx这种方式填写,然后再导出时只需读取模板然后用相应的数据替换其中的{xxx}这种方式填写, 然后再导出时只需读取模板然后用相应的数据替换其中的xxx这种方式填写,然后再导出时只需读取模板然后用相应的数据替换其中的{xxx}即可.

   一,简单模板导出(不含图片, 不含表格循环) 1, 新建一个word文档, 输入如下类容: 2, 将该word文件另存为xml格式(注意是另存为,不是直接改扩展名) 3, 将xml文件的扩展名直接改为ftl 4, 用java代码完成导出(需要导入freemarker.jar)

Java代码:

@Test
public void test(){
Map<String,Object> dataMap = new HashMap<String, Object>();
try {
//编号
dataMap.put(“id”, “123456”);
//日期
dataMap.put(“date”, new SimpleDateFormat(“yyyy年MM月dd日”).format(new SimpleDateFormat(“yyyy-MM-dd”).parse(“2018-09-19”)));
//附件张数
dataMap.put(“number”, 1);
//受款人
dataMap.put(“payee”, “张三”);
//付款用途
dataMap.put(“use_of_payment”, “test”);
//大写金额
dataMap.put(“capitalization_amount”, MoneyUtils.change(100.20));
//小写金额
dataMap.put(“lowercase_amount”, “100”);
//Configuration 用于读取ftl文件
Configuration configuration = new Configuration(new Version(“2.3.0”));
configuration.setDefaultEncoding(“utf-8”);

        /*** 以下是两种指定ftl文件所在目录路径的方式,注意这两种方式都是* 指定ftl文件所在目录的路径,而不是ftl文件的路径*///指定路径的第一种方式(根据某个类的相对路径指定)

// configuration.setClassForTemplateLoading(this.getClass(), “”);

        //指定路径的第二种方式,我的路径是C:/a.ftlconfiguration.setDirectoryForTemplateLoading(new File("c:/"));//输出文档路径及名称File outFile = new File("D:/报销信息导出.doc");//以utf-8的编码读取ftl文件Template template = configuration.getTemplate("报销信息导出.ftl", "utf-8");Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "utf-8"), 10240);template.process(dataMap, out);out.close();} catch (Exception e) {e.printStackTrace();}
}

注意:我使用的 freemarker 版本为2.3.28,Configuration对象不推荐直接new Configuration(),仔细看Configuration.class文件会发现,推荐的是 Configuration(Version incompatibleImprovements) 这个构造方法,具体这个构造方法里面传的就是Version版本类,而且版本号不能低于2.3.0

  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<String,String> dataMap = new HashMap<String,String>();
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是集合中的每个元素, 类似<c:forEach items='userList' var='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{  //构造数据  Map<String,List> dataMap = new HashMap<String,List>();  List<User> list = new ArrayList<User>();  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:  <STRING_LITERAL> ...  <RAW_STRING> ...  "false" ...  "true" ...  <INTEGER> ...  <DECIMAL> ...  "." ...  "+" ...  "-" ...  "!" ...  "[" ...  "(" ...

这是由于在写xxx表达式的时候,xxx与其前方的文字样式不一致,在另存为xml后你可以搜索一下"{xxx}表达式的时候, xxx与其前方的文字样式不一致, 在另存为xml后你可以搜索一下 "xxx表达式的时候,xxx与其前方的文字样式不一致,在另存为xml后你可以搜索一下"{" , 会发现如下图这种情况:

由于xxx中的xxx格式与其前方的文字不一致,那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签,所以在word中看似写的是{xxx}中的xxx格式与其前方的文字不一致, 那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签, 所以在word中看似写的是xxx中的xxx格式与其前方的文字不一致,那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签,所以在word中看似写的是{xxx}实际上转为xml后变成了<w:color...>xxx</w:color>,这样这个el表达式中的标签就解析不了报错了,可以去掉{<w:color ...>xxx</w:color>},这样这个el表达式中的标签就解析不了报错了, 可以去掉<w:color...>xxx</w:color>,这样这个el表达式中的标签就解析不了报错了,可以去掉{}内部的标签只留下xxx或者删掉 “${” 和 "}"然后给xxx加上el表达式都可以解决此问题.

五, javaWeb中利用response导出(注意编码问题,防止中文乱码)

 Map<String,String> dataMap = new HashMap<String,String>();  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 export word template

原文链接:https://blog.csdn.net/zhangzhangjava/article/details/52937159

最近需要做一个导出word的功能, 在网上搜了下, 有用POI,JXL,iText等jar生成一个word文件然后将数据写到该文件中,API非常繁琐而且拼出来的样式也不美观,于是选择了另一种方式----feemarker基于word模板的导出方式, 这种方式非常简单而且导出的样式美观, 其原理就是先做一个word模板, 该模板中变量数据用xxx这种方式填写,然后再导出时只需读取模板然后用相应的数据替换其中的{xxx}这种方式填写, 然后再导出时只需读取模板然后用相应的数据替换其中的xxx这种方式填写,然后再导出时只需读取模板然后用相应的数据替换其中的{xxx}即可.

   一,简单模板导出(不含图片, 不含表格循环) 1, 新建一个word文档, 输入如下类容: 2, 将该word文件另存为xml格式(注意是另存为,不是直接改扩展名) 3, 将xml文件的扩展名直接改为ftl 4, 用java代码完成导出(需要导入freemarker.jar)

Java代码:

@Test
public void test(){
Map<String,Object> dataMap = new HashMap<String, Object>();
try {
//编号
dataMap.put(“id”, “123456”);
//日期
dataMap.put(“date”, new SimpleDateFormat(“yyyy年MM月dd日”).format(new SimpleDateFormat(“yyyy-MM-dd”).parse(“2018-09-19”)));
//附件张数
dataMap.put(“number”, 1);
//受款人
dataMap.put(“payee”, “张三”);
//付款用途
dataMap.put(“use_of_payment”, “test”);
//大写金额
dataMap.put(“capitalization_amount”, MoneyUtils.change(100.20));
//小写金额
dataMap.put(“lowercase_amount”, “100”);
//Configuration 用于读取ftl文件
Configuration configuration = new Configuration(new Version(“2.3.0”));
configuration.setDefaultEncoding(“utf-8”);

        /*** 以下是两种指定ftl文件所在目录路径的方式,注意这两种方式都是* 指定ftl文件所在目录的路径,而不是ftl文件的路径*///指定路径的第一种方式(根据某个类的相对路径指定)

// configuration.setClassForTemplateLoading(this.getClass(), “”);

        //指定路径的第二种方式,我的路径是C:/a.ftlconfiguration.setDirectoryForTemplateLoading(new File("c:/"));//输出文档路径及名称File outFile = new File("D:/报销信息导出.doc");//以utf-8的编码读取ftl文件Template template = configuration.getTemplate("报销信息导出.ftl", "utf-8");Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "utf-8"), 10240);template.process(dataMap, out);out.close();} catch (Exception e) {e.printStackTrace();}
}

注意:我使用的 freemarker 版本为2.3.28,Configuration对象不推荐直接new Configuration(),仔细看Configuration.class文件会发现,推荐的是 Configuration(Version incompatibleImprovements) 这个构造方法,具体这个构造方法里面传的就是Version版本类,而且版本号不能低于2.3.0

  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<String,String> dataMap = new HashMap<String,String>();
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是集合中的每个元素, 类似<c:forEach items='userList' var='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{  //构造数据  Map<String,List> dataMap = new HashMap<String,List>();  List<User> list = new ArrayList<User>();  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:  <STRING_LITERAL> ...  <RAW_STRING> ...  "false" ...  "true" ...  <INTEGER> ...  <DECIMAL> ...  "." ...  "+" ...  "-" ...  "!" ...  "[" ...  "(" ...

这是由于在写xxx表达式的时候,xxx与其前方的文字样式不一致,在另存为xml后你可以搜索一下"{xxx}表达式的时候, xxx与其前方的文字样式不一致, 在另存为xml后你可以搜索一下 "xxx表达式的时候,xxx与其前方的文字样式不一致,在另存为xml后你可以搜索一下"{" , 会发现如下图这种情况:

由于xxx中的xxx格式与其前方的文字不一致,那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签,所以在word中看似写的是{xxx}中的xxx格式与其前方的文字不一致, 那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签, 所以在word中看似写的是xxx中的xxx格式与其前方的文字不一致,那么在生成xml时,就会有一些修饰xxx样式的标签,例如修饰xxx的字体,颜色等的标签,所以在word中看似写的是{xxx}实际上转为xml后变成了<w:color...>xxx</w:color>,这样这个el表达式中的标签就解析不了报错了,可以去掉{<w:color ...>xxx</w:color>},这样这个el表达式中的标签就解析不了报错了, 可以去掉<w:color...>xxx</w:color>,这样这个el表达式中的标签就解析不了报错了,可以去掉{}内部的标签只留下xxx或者删掉 “${” 和 "}"然后给xxx加上el表达式都可以解决此问题.

五, javaWeb中利用response导出(注意编码问题,防止中文乱码)

 Map<String,String> dataMap = new HashMap<String,String>();  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模板导出相关推荐

  1. Java使用模板导出带图片word文档

    之前已经写过一个 Java使用模板导出Word文档 由于系统升级,模板得添加图片,这个时候遇到了一个比较恶心的问题特地发文记录一下. 先把老模板翻出来 **.ftl 然后右键用wps或者office打 ...

  2. java按照模板导出pdf或者word

    一.java按照模板导出pdf (一)制作模板 1.在word里制作模板 因为PDF常用的软件不支持编辑,所以先用Word工具,如WPS或者Office新建一个空白Word文档,里面制作出自己想要的样 ...

  3. SpringBoot集成文件 - 如何基于POI-tl和word模板导出庞大的Word文件?

    前文我们介绍了通过Apache POI通过来导出word的例子:那如果是word模板方式,有没有开源库通过模板方式导出word呢?poi-tl是一个基于Apache POI的Word模板引擎,也是一个 ...

  4. 8、jeecg 笔记之 自定义word 模板导出(一)

    8.jeecg 笔记之 自定义word 模板导出(一) 1.前言 jeecg 中已经自带 word 的导出导出功能,其所使用的也是 easypoi,尽管所导出的 word 能满足大部分需求, 但总是有 ...

  5. SpringBoot+EasyPOI word模板导出,含多张图片

    这几天客户提出了新的需求,要求记录要能够导出word,并且里面包含的图片也要导出来,这里借用EasyPOI来进行操作. 参考文章:https://blog.csdn.net/qq_34752942/a ...

  6. java使用模板导出Excel表格

    java使用模板导出Excel表格 文章目录 java使用模板导出Excel表格 模板示例 一.引入依赖 二.使用步骤 1.添加工具类 2.导出Excel 模板示例 一.引入依赖 <depend ...

  7. poi-tl,根据word模板导出word(表格行循环,表格无表头的情况)

    最近项目里要做一个根据客户提供的word模板导出word的功能,方法有很多,比如easyPoi(对word的支持并不是很好),freeMark(太麻烦不想研究),以及poi-tl, 最后研究了半天发现 ...

  8. Net Core DocXCore 实现word模板导出

    实际工作中,往往有这样的需求,需要导出word,还有各种各样的样式,于是有了word模板导出. 实现以下几个需求: 1.表单导出 2.表格导出 3.表单表格混合导出 4.实际用例测试 解决方案: 实现 ...

  9. 在docxtemplater 和 open-docxtemplater-image-module按word模板导出图片 遇到个坑记录下

    在docxtemplater 和 open-docxtemplater-image-module按word模板导出图片 遇到个坑记录下 主要是一直提示 Cannot read property 'pa ...

  10. Java按照word模板导出、下载文档

    在日常的开发工作中,我们时常会遇到导出Word文档报表的需求,比如公司的财务报表.医院的患者统计报表.电商平台的销售报表等等. 导出Word方式多种多样,通常有以下几种方式: 1. 使用第三方Java ...

最新文章

  1. 实践重于理论——创建一个监控程序探测WCF的并发处理机制
  2. 牛客网《BAT面试算法精品课》学习笔记
  3. 状态机-重叠序列检测
  4. 《数据结构与算法分析-C语言描述》习题2.6
  5. java用一个方法对变量初始化_java中怎么给变量初始化?
  6. python从oracle提取数据库_Python编程实战之Oracle数据库操作示例
  7. ES5实现ES6的一些方法-call,bind,is,promise
  8. pandas 多线程处理数据框
  9. discuz uc_server 配置登录
  10. 技术架构图-软件研发流程
  11. STC15单片机功能和应用电路
  12. 天涯“大鹏金翅明王”语录
  13. SDI Over IP相关标准及技术简介
  14. Android开发通知栏的那些事
  15. 数据处理之衡量数据远近的多种距离公式
  16. STM32电路原理图
  17. 第五章(第四节)论路由器重置
  18. Java时间日期类处理(LocalDateTime、LocalDate相关操作、获取周几、工作日休息日节假日判定)
  19. 互联网基础与应用复习
  20. i5 1135g7参数 i5 1135g7核显性能

热门文章

  1. VMware安装deepin
  2. VMWare安装Deepin系统
  3. Python 代理验证程序(For 无忧代理) 多线程版
  4. 【机器学习理论】换底公式--以e,2,10为底的对数关系转化
  5. 我们和蹦迪博主聊了聊,哪家大厂人在夜店最受欢迎?
  6. 14 ABSOLUTE评估肿瘤纯度
  7. 肿瘤 材料 计算机模拟,PeerJ: 计算机模拟揭示癌症多样性
  8. 转 我奋斗了18年,不是为了和你一起喝咖啡!
  9. Win7 远程桌面限制IP
  10. 猿辅导 iOS 精准测试实践 - Objective-C 与 Swift 混编工程精准测试探索