java使用freemaker 导出word 包含分页,表格循环,word改xml格式化

在平时项目开发中,相信或多或少的都遇到过word导出的需求,这里整理一个比较全面的java word导出方法,希望可以跟大家一起交流学习。
  • 创建word模板

  • 1.1 新建一个word模板,并修改字段

    在这里插入图片描述

    1. 2 将word文件另存为xml 格式,打开xml 格式文件,你会发现你刚刚编写的会变成这样。

      这里推荐一个比较好用的编辑器 sublime 具体教程可以参考这一篇sublime

      安装好之后,修改刚刚的xml文件。
      在这里插入图片描述
      把需要更改的变量都替换成${xxx}.改完后,把xml文档重命名成后缀为.ftl 的freemaker文件。

      1.3 列表和分页
      分页标签为 <w:p><w:r><w:br w:type="page"/></w:r></w:p>
      在word xml 格式中,<w:body></w:body> 表示的就是整个文本内容。我这里的业务需求是不同的类型的内容分页打印,既然<w:body></w:body> 表示整个文本内容,那么就可以在模板中写多个<w:body></w:body> ,不同的分类显示不同的word模板内容。我这里分了三类,就有三个不同的分页。
      在这里插入图片描述
      列表循环
      如果你的模板中包含表格, 找到 <w:tbl></w:tbl> 标签,使用<#list></#list>指令 循环你的list<对象> 数据,然后使用 ${} 取到相应的值。
      在这里插入图片描述
      自此,ftl 模板部分全部写完,将ftl 文件放入你电脑的某个位置,接下来可以开开心心的写java代码部分。

2.java后台编写

思路 :1.编写读取ftl 模板以及参数的方法
2.查询你需要填入word的数据,放入map。注意map的key一定要跟你在ftl文件中填写的字段对应。
话不多说,直接上代码。

package cn.ys.common.utils;

import java.io.BufferedWriter;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStreamWriter;
import java.io.PrintWriter;
import java.io.Writer;
import java.net.HttpURLConnection;
import java.net.URL;
import java.net.URLEncoder;
import java.util.Date;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.commons.lang.StringUtils;
import org.apache.poi.POIXMLDocument;
import org.apache.poi.POIXMLTextExtractor;
import org.apache.poi.hwpf.HWPFDocument;
import org.apache.poi.hwpf.usermodel.Range;
import org.apache.poi.openxml4j.opc.OPCPackage;
import org.apache.poi.xwpf.extractor.XWPFWordExtractor;
import org.apache.poi.xwpf.usermodel.XWPFDocument;

import sun.misc.BASE64Encoder;
import cn.ys.common.web.HtmlMessage;

import com.google.common.collect.Maps;

import freemarker.template.Configuration;
import freemarker.template.Template;
import freemarker.template.TemplateException;

public class WordUtils {

	private WordUtils() {
		throw new AssertionError();
	}
	/**
	 *
	 * @param templateFolder 你的ftl文件位置
	 * @param ftlFile  你的ftl 文件名
	 * @param map  需要渲染 数据
	 * @param localPath  生成word文档的位置    
	 * @param title  word文档名
	 * @return
	 * @throws IOException
	 */
	public static String exportWord(String templateFolder, String ftlFile, Map map, String localPath, String title) throws IOException {
		Configuration configuration = new Configuration();
		configuration.setDefaultEncoding("utf-8");
		configuration.setClassicCompatible(true);
		configuration.setDirectoryForTemplateLoading(new File(templateFolder)); //读取ftl 模板位置

		try {
			String datePath = DateUtils.getNowTime("yyyy/MM/dd");
			//使用时间为文件夹生成word生成的位置
			String path = localPath + "/word/" + datePath; 
			//定义word文档名称
			String fileName = title + ".doc";  
			//创建文件夹
			File outFile = new File(path + "/" + fileName);
			if (!outFile.getParentFile().exists()) {
				outFile.getParentFile().mkdirs();
			}
			//读取 模板内容
			Template template = configuration.getTemplate(ftlFile, "utf-8");

			Writer out = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(outFile), "UTF-8"));

			template.process(map, out);

			out.flush();
			out.close();
			
			return path + "/" + fileName;
		} catch (TemplateException e) {
			e.printStackTrace();
		}
		return "";
	}


}

自此,你的doc文件实际上已经生成在你指定的文件夹了。 接下来如果让他能够使用文浏览器下载,只需要去读取你本地的这个文件就行。`

  @SuppressWarnings("unchecked")
    @RequestMapping(value = "/word/download/{id}")
    public void exportSupplier(@PathVariable("id") Integer gysbaId, HttpServletResponse response, HttpServletRequest request) {
        try {
 				//  注意这里的参数,根据你自己业务出入,参数说明上面已经注明!
                String downLoadPath = WordUtils.exportWord(templateFolder,ftlName,map,root,title);
                String fileName = map.get("message").toString();

                File file = new File(downLoadPath);
                InputStream inputStream;
                inputStream = new BufferedInputStream(new FileInputStream(file));
                byte[] buffer = new byte[inputStream.available()];
                inputStream.read(buffer);
                inputStream.close();

                response.reset();

                String userAgent = request.getHeader("user-agent").toLowerCase();
                if (userAgent.contains("msie") || userAgent.contains("like gecko")) {
                    fileName = URLEncoder.encode(fileName, "UTF-8");
                } else {
                    fileName = new String(fileName.getBytes("UTF-8"), "iso-8859-1");
                }

                response.setHeader("Content-disposition", String.format("attachment; filename=\"%s\"", fileName));

                response.addHeader("Content-Length", "" + file.length());

                response.setContentType("application/x-download");
                os.write(buffer);// 输出文件
                os.flush();
                os.close();
         
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

好的,大功告成!如果发现有什么问题,希望各位大佬指出,共同学习。

love & peace

  • 7
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
在Web应用中,有时需要按照固定的模板将数据导出Word,如流程审批单,在流程处理完成后将处理过程按照流程单的要求导出,有时程序中需要实现生成标准Word文档,要求能够打印,并且保持页面样式不变,常见的方案有POI、iText、JACOB、JSP几种方式,POI读取Word文档比较适合、对于生成文档样式比较难控制,iText操作Excel还可以,对Word的操作功能有限,JACOB操作Word实现复杂,并且无法将服务部署到Linux平台,要求安装office,对于实现固定格式的报表实现困难,对于JSP直接输出方式样式控制难。 Word从2003开始支持XML格式,用XML+Freemarder还做就很简单了,大致的思路是先用office2003或者2007编辑好 word的样式,然后另存为xml,将xml翻译为FreeMarker模板,最后用java来解析FreeMarker模板并输出Doc。经测试这样方式生成的word文档完全符合office标准,样式、内容控制非常便利,打印也不会变形,生成的文档和office中编辑文档完全一样。具体实现过程如下: 1、 首先用office【版本要2003以上,以下的不支持xml格式】编辑文档的样式,将需要动态填充的内容使用Freemarker标签替换:Word文档样式如下: 2、 将Word文档另存为XML格式,将后缀名“xml修改为“ftl” 3、 使用Freemarker填充内容,代码如下: [java] view plaincopyprint? 1. package com.test.freemarker.report; 2. 3. 4. 5. import java.io.BufferedWriter; 6. 7. import java.io.File; 8. 9. import java.io.FileOutputStream; 10. 11. import java.io.IOException; 12. 13. import java.io.OutputStreamWriter; 14. 15. import java.io.Writer; 16. 17. import java.util.HashMap; 18. 19. import java.util.Map; 20. 21. 22. 23. import freemarker.template.Configuration; 24. 25. import freemarker.template.Template; 26. 27. import freemarker.template.TemplateException; 28. 29. 30. 31. public class DocumentHandler { 32. 33. private Configuration configuration = null; 34. 35. 36. 37. public DocumentHandler() { 38. 39. configuration = new Configuration(); 40. 41. configuration.setDefaultEncoding("utf-8"); 42. 43. } 44. 45. 46. 47. public void createDoc() { 48. 49. // 要填入模本的数据文件 50. 51. Map dataMap = new HashMap(); 52. 53. getData(dataMap); 54. 55. // 设置模本装置方法和路径,FreeMarker支持多种模板装载方法。可以重servlet,classpath,数据库装载, 56. 57. // 这里我们的模板是放在com.havenliu.document.template包下面 58. 59.
Freemarker 是一种模板引擎,可用于动态生成文档,包括导出 Word 文档。在 Word 中拆分表格单元格是一个常见的需求,在使用 Freemarker 导出 Word 表格时,也可以通过一些技巧来实现。 首先,我们需要将表格数据传递给 Freemarker 模板。可以通过一个二维数组或者一个包含多个 Map 的 List 来表示表格的数据。每个 Map 表示表格的一行数据,其中的 key 表示列的标题,value 表示该单元格的内容。 然后,在 Freemarker 模板中,可以使用表格标签来生成表格使用嵌套的 #list 指令来遍历行数据,使用嵌套的 #items 指令来遍历列数据。可以通过设定参数来控制表格的样式,例如边框、背景颜色等。 接下来,需要判断是否需要拆分单元格。可以在模板中使用 #if 指令来判断某个条件是否满足,例如某个单元格的内容是否符合拆分的条件。如果需要拆分,则可以使用 colspan 和 rowspan 参数来控制单元格的合并和拆分。 需要注意的是,在拆分单元格时,应该确保表格的结构仍然正确,不会造成表格乱码或无法显示的问题。可以通过在表格标签中增加条件判断、控制行和列的合并等方式来调整表格结构。 最后,通过 Freemarker 的模板引擎将数据应用于模板,并将生成的 Word 文档导出保存即可。 总结起来,使用 Freemarker 导出 Word 表格并拆分单元格,需要将表格数据传递给 Freemarker 模板,通过判断条件来决定是否拆分单元格,并注意保持表格结构的正确性。这样就可以生成带有拆分单元格的 Word 文档。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值