Freemaker基于word模板动态导出压缩文件汇总整理

Freemaker基于word模板动态导出单个文件思路和代码详情见连接:

https://www.cnblogs.com/lsy-blogs/p/9243281.html

核心思路如下:

1、service中写方法,查询需要导出的结果list集合数据;

2、控制层中调用service方法获取结果集合数据,调用freemaker获取对应的word文件到临时目录下;

3、利用流处理,将临时目录下导出的每一个word写入到压缩文件中去,利用流输出到浏览器中,实现压缩文件下载;

4、最终将各种流关闭,并将产生的临时word文件删除;

核心是工具类和控制层的调用,核心代码如下:

控制层代码:

@RequestMapping(params = "exportBatchResultScoreWordZip")
public void exportBatchResultScoreWordZip(String ids, HttpServletRequest request,
HttpServletResponse response, DataGrid dataGrid, ModelMap modelMap) {
//获取要批量导出的result_score表的主键id数组
String[] resultScoreIds = ids.split(",");
// 调用service中的方法,获取批量导出word导出需要的目录主表、字段表信息结果集合
List<ProvinceCityResultOneVO> resultOneVOList = tBAssessResultScoreService.getWrodResultListByResultScoreIds(resultScoreIds);
List<Map<String, Object>> wordResultDataMapList = new ArrayList<Map<String, Object>>();
String departLevel = "";
String resultRarFileName = "";
for(int i = ; i < resultOneVOList.size(); i++) {
ProvinceCityResultOneVO resultOneVO = resultOneVOList.get(i);
String appraiseYear = resultOneVO.getAppraiseYear();
String departName = resultOneVO.getDepartName();
String departType = resultOneVO.getDepartType();
String fileName = appraiseYear+"年"+departName+"结果";
Map<String, Object> assessResultMap = new HashMap<String, Object>();
assessResultMap.put("assessResul", resultOneVO);
assessResultMap.put("fileName", fileName);
String templateName = "";
if("".equals(departType)) {
templateName = "provinceResultScoreBatchWordRar.xml";
}else if("".equals(departType)) {
templateName = "cityResultScoreBatchWordRar.xml";
}
if(i==) {
departLevel = departType;
}
assessResultMap.put("templateName", templateName);
wordResultDataMapList.add(assessResultMap);
}
if(StringUtil.isNotEmpty(departLevel)) {
Date nowDate = new Date();
SimpleDateFormat simpleDateFormat = new SimpleDateFormat("yyyyMMddHHmmss");
String nowDateStr = simpleDateFormat.format(nowDate);
if("".equals(departLevel)) {
//省批量导出压缩文件名称赋值
resultRarFileName = "结果_"+nowDateStr;
}else if("".equals(departLevel)) {
//市批量导出压缩文件名称赋值
resultRarFileName = "结果_"+nowDateStr;
}
}
String resultFileType = "zip";
try {
WordUtils.exportMillCertificateWordZip(request, response, wordResultDataMapList,resultRarFileName,resultFileType);
} catch (IOException e) {
e.printStackTrace();
}
}

word导出工具类代码:

package com.jeecg.assessResultScore.utils;

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStreamWriter;
import java.io.Writer;
import java.net.URLEncoder;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.UUID;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream; import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse; import org.jeecgframework.core.util.StringUtil; import com.jeecg.assessResultScore.vo.ProvinceCityResultOneVO; import freemarker.template.Configuration;
import freemarker.template.Template; public class WordUtils { // 配置信息,代码本身写的还是很可读的,就不过多注解了
private static Configuration configuration = null;
// 这里注意的是利用WordUtils的类加载器动态获得模板文件的位置
private static final String templateFolder = WordUtils.class.getClassLoader().getResource("../../").getPath()
+ "export/template/";
static {
configuration = new Configuration();
configuration.setDefaultEncoding("utf-8");
try {
configuration.setDirectoryForTemplateLoading(new File(templateFolder));
} catch (IOException e) {
e.printStackTrace();
}
} private WordUtils() {
throw new AssertionError();
} /**
* 下载单个word文件
* @param request 请求
* @param response 响应
* @param map word结果数据
* @param fileName 结果文件名称(不需要带后缀的)
* @param wordXmlName word模板名称
* @throws IOException
*/
@SuppressWarnings({ "rawtypes", "unchecked" })
public static void exportMillCertificateWord(HttpServletRequest request, HttpServletResponse response, Map map,String fileName,String wordXmlName)
throws IOException {
Template freemarkerTemplate = configuration.getTemplate(wordXmlName);
File file = null;
InputStream fin = null;
ServletOutputStream out = null;
try {
// 调用工具类的createDoc方法生成Word文档
file = createDoc(map, freemarkerTemplate);
fin = new FileInputStream(file);
//根据不同浏览器,对fileName进行不同的编码
String userAgent = request.getHeader("user-agent").toLowerCase();
if (userAgent.contains("msie") || userAgent.contains("like gecko") ) {
// win10 ie edge 浏览器 和其他系统的ie
fileName = URLEncoder.encode(fileName, "UTF-8");
} else {
// fe
fileName = new String(fileName.getBytes("utf-8"), "iso-8859-1");
}
response.setCharacterEncoding("utf-8");
response.setContentType("application/msword");
// 设置浏览器以下载的方式处理该文件名
fileName = fileName+".doc";
response.setHeader("Content-Disposition",
"attachment;filename="+fileName); out = response.getOutputStream();
byte[] buffer = new byte[]; // 缓冲区
int bytesToRead = -;
// 通过循环将读入的Word文件的内容输出到浏览器中
while ((bytesToRead = fin.read(buffer)) != -) {
out.write(buffer, , bytesToRead);
}
} finally {
if (fin != null) {
fin.close();
}
if (out != null) {
out.flush();
out.close();
}
if (file != null) {
file.delete(); // 删除临时文件
}
}
} /**
* 下载批量word文件到一个rar压缩文件中去
* @param request 请求
* @param response 响应
* @param mapList 存放多个word文件的信息:fileName:文件名(不需要带后缀的)、templateName:模板名称、assessResul:数据结果信息的集合
* @param resultRarFileName 最终下载的压缩文件名称
* @param resultFileType 最终下载的压缩文件格式
* @throws IOException
*/
public static void exportMillCertificateWordZip(HttpServletRequest request, HttpServletResponse response, List<Map<String, Object>> mapList,String resultRarFileName,String resultFileType)
throws IOException {
File[] docTempleFiles = new File[mapList.size()];
String[] docTempleFileName = new String[mapList.size()];
ZipOutputStream zipOutputStream = null;
try {
//遍历结果数据集合,将word模板生成的文件保存在临时文件夹中,文件记录在数组中;
for(int i = ; i < mapList.size(); i++) {
String templateName = mapList.get(i).get("templateName").toString();
if(StringUtil.isNotEmpty(templateName)) {
File docFile = new File(templateFolder+UUID.randomUUID().toString()+".doc");
// 创建 FileInputStream 对象
String fileName = mapList.get(i).get("fileName").toString();
ProvinceCityResultOneVO resultOneVO = (ProvinceCityResultOneVO)mapList.get(i).get("assessResul");
Map<String, Object> resultOneVOMap = new HashMap<String, Object>();
resultOneVOMap.put("assessResul", resultOneVO);
// 调用工具类的createDoc方法生成Word文档
Template freemarkerTemplate = configuration.getTemplate(templateName);
docFile = createZipDoc(resultOneVOMap, freemarkerTemplate,docFile);
//将doc文件放到数组中去
docTempleFiles[i] = docFile;
//将doc文件中文名称放到文件名数组中去
docTempleFileName[i] = fileName;
}
}
//根据不同浏览器,对fileName进行不同的编码
String userAgent = request.getHeader("user-agent").toLowerCase();
if (userAgent.contains("msie") || userAgent.contains("like gecko") ) {
// win10 ie edge 浏览器 和其他系统的ie
resultRarFileName = URLEncoder.encode(resultRarFileName, "UTF-8");
} else {
// fe
resultRarFileName = new String(resultRarFileName.getBytes("utf-8"), "iso-8859-1");
}
response.setCharacterEncoding("utf-8");
response.setContentType("application/octet-stream");
// 设置浏览器以下载的方式处理该文件名
resultRarFileName = resultRarFileName+"."+resultFileType;
response.setHeader("Content-Disposition",
"attachment;filename="+resultRarFileName);
zipOutputStream = new ZipOutputStream(response.getOutputStream());
//遍历文件数组,将文件压缩到zip格式的压缩文件中去,
for(int i = ; i < docTempleFiles.length; i++) {
File docFile = docTempleFiles[i];
FileInputStream fileInputStream = new FileInputStream(docFile);
ZipEntry entry = new ZipEntry(docTempleFileName[i]+".doc");
zipOutputStream.putNextEntry(entry);
int len = -;
byte[] buffer = new byte[];
while ((len = fileInputStream.read(buffer)) != -) {
zipOutputStream.write(buffer, , len);
}
zipOutputStream.closeEntry();
fileInputStream.close();
}
zipOutputStream.flush();
} finally {
if (zipOutputStream != null) {
zipOutputStream.flush();
zipOutputStream.close();
}
for(int i = ; i < docTempleFiles.length; i++) {
docTempleFiles[i].delete();//删除临时doc文件
}
}
} private static File createDoc(Map<String, Object> dataMap, Template template) {
String name = "test.doc";
File f = new File(name);
Template t = template;
try {
// 这个地方不能使用FileWriter因为需要指定编码类型否则生成的Word文档会因为有无法识别的编码而无法打开
Writer w = new OutputStreamWriter(new FileOutputStream(f), "utf-8");
t.process(dataMap, w);
w.close();
} catch (Exception ex) {
ex.printStackTrace();
throw new RuntimeException(ex);
}
return f;
} private static File createZipDoc(Map<String, Object> dataMap, Template template,File f) {
Template t = template;
try {
// 这个地方不能使用FileWriter因为需要指定编码类型否则生成的Word文档会因为有无法识别的编码而无法打开
Writer w = new OutputStreamWriter(new FileOutputStream(f), "utf-8");
t.process(dataMap, w);
w.close();
} catch (Exception ex) {
ex.printStackTrace();
throw new RuntimeException(ex);
}
return f;
}
}

Freemaker基于word模板动态导出压缩文件汇总整理的更多相关文章

  1. Freemaker基于word模板动态导出汇总整理

    Freemaker基于word模板动态导出汇总整理 一.使用的jar包: 二.Word模板动态导出的基本思路: 1.首先通过自己在word中创建好需要导出的word文本+表格的模板,模板中需要填写内容 ...

  2. JSP利用freemarker生成基于word模板的word文档

    利用freemarker生成基于word模板的word文档 freemarker简介 FreeMarker是一个用Java语言编写的模板引擎,它基于模板来生成文本输出.FreeMarker与Web容器 ...

  3. C# 通过word模板动态生成Word

    object oMissing = System.Reflection.Missing.Value; Word._Application oWord = new Word.Application(); ...

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

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

  5. Excel模板导出之动态导出

    说明 目前Magicodes.IE已支持Excel模板导出时使用JObject.Dictionary和ExpandoObject来进行动态导出,具体使用请看本篇教程. 本功能的想法.部分实现初步源于a ...

  6. java动态导出PDF(利用itext)

    项目基于ssm框架,使用itext动态导出pdf文件: 1.引入两个jar包:itextpdf-5.5.5.jar.itext-asian-5.2.0.jar 说明: 1.itextpdf-5.5.5 ...

  7. 利用模板导出文件(二)之jacob利用word模板导出word文件(Java2word)

    https://blog.csdn.net/Fishroad/article/details/47951061?locationNum=2&fps=1 先下载jacob.jar包.解压后将ja ...

  8. poi导出word模板项目实例(一个文件)

    在页面上填写值,然后导出到word模板中,并把页面上的值带到模板中,也就是导出word文档,提前有word 的模板形式, 1.jsp 页面   <table class="formTa ...

  9. Java 使用模板生成 Word 文件---基于 Freemarker 模板框架

    Java项目引入 Freemarker 插件自行完成. 步骤如下: .编写 Word 模板,并将模板中要用代码动态生成数据用 Freemarker 变量取代,即${变量名},如${username}: ...

随机推荐

  1. Mysql遍历大表(Mysql大量数据读取内存溢出的解决方法)

    mysql jdbc默认把select的所有结果全部取回,放到内存中,如果是要遍历很大的表,则可能把内存撑爆. 一种办法是:用limit,offset,但这样你会发现取数据的越来越慢,原因是设置了of ...

  2. rancher中级(二)(rancher中添加证书及操作虚拟主机)

    制作一个ssl证书 首先了解关于ssl证书的背景知识:http://www.cnblogs.com/zxj015/p/4458066.html SSL证书包括: 1,CA证书,也叫根证书或者中间级证书 ...

  3. jdk代理

    接口: public interface IUserService { public void saveUser(String username,String password); public vo ...

  4. Storm概念学习系列之storm简介

    不多说,直接上干货! storm简介 Storm 是 Twitter 开源的.分布式的.容错的实时计算系统,遵循 Eclipse Public License1.0. Storm 通过简单的 API ...

  5. 深入理解C#中的IDisposable接口(转)

    转自:https://www.cnblogs.com/wyt007/p/9304564.html 写在前面 在开始之前,我们需要明确什么是C#(或者说.NET)中的资源,打码的时候我们经常说释放资源, ...

  6. pat1085. Perfect Sequence (25)

    1085. Perfect Sequence (25) 时间限制 300 ms 内存限制 65536 kB 代码长度限制 16000 B 判题程序 Standard 作者 CAO, Peng Give ...

  7. sql server sql语句

    添加联合唯一索引 create unique index 索引名 on 表名(列名1,列名2……)

  8. Unity3d中使用assetbundle

    1.导出assetbundle: ①单个资源导出成assetbundle: ②多个资源导出成一个assetbundle: 2.读取assetbundle: ①加载到内存: ②解压为具体资源. 1.导出 ...

  9. Java 编码规范有感

    应小组要求,开发测试都需要考阿里编码规范,因此,相当于是突击了一下关于编码规范方面的知识,目前做的项目后期需要进行项目迁移,数据迁移,功能迁移... 各种迁移... 阿里巴巴编码规范(Java)考试地 ...

  10. Erlang 001--开篇

    有段时间没有更新博客了,最近稍微接触了下一门相对小众的语言Erlang,个人感觉学习一段时间有必要总结总结,本文作为该系列的开篇,仅仅列举一些与Java的一些不同点和个人对Erlang的一些主观印象, ...