我看了很多网上的demo,先生成ZIP压缩文件,然后再下载。

我这里是生成ZIP文件流 进行下载。(核心代码没多少,就是一些业务代码)

@RequestMapping(value = "/")
public ResponseEntity<byte[]> downloadInterviewFile() throws Exception {
// 根据面试官主键编码 下载文件
List<InterviewFile> interviewFiles = this.interviewFileService.getAll(interviewfile); ByteArrayOutputStream byteOutPutStream = null; if (!interviewFiles.isEmpty()) {
//单个文件名称
String interviewFileName = "";
//文件后缀
String fileNameSuffix = "";
//创建一个集合用于 压缩打包的参数
List<Map<String, String>> parms = new ArrayList<>();
//创建一个map集合
Map<String, String> fileMaps =null;
//得到存储文件盘符 例 D:
String root = properties.getValue("upload.root");
//创建一个字节输出流
byteOutPutStream = new ByteArrayOutputStream(); for (InterviewFile file : interviewFiles) { fileMaps = new HashMap<>(); interviewFileName = file.getFileName(); fileNameSuffix = file.getFileSuffix();
//将单个存储路径放入
fileMaps.put("filePath", root.concat(file.getFilePath()));
//将单个文件名放入
fileMaps.put("fileName", interviewFileName.concat("." + fileNameSuffix));
//放入集合
parms.add(fileMaps); }
//压缩文件
FileUtils.batchFileToZIP(parms, byteOutPutStream);
} HttpHeaders headers = new HttpHeaders(); String fileName = null;
try {
fileName = new String("附件.zip".getBytes("UTF-8"), "ISO-8859-1");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); headers.setContentDispositionFormData("attachment", fileName);// 文件名称 ResponseEntity<byte[]> responseEntity = new ResponseEntity<byte[]>(byteOutPutStream.toByteArray(), headers, HttpStatus.CREATED); return responseEntity; }

工具类

@RequestMapping(value = "/")
public ResponseEntity<byte[]> downloadInterviewFile() throws Exception {
// 根据面试官主键编码 下载文件
List<InterviewFile> interviewFiles = this.interviewFileService.getAll(interviewfile); ByteArrayOutputStream byteOutPutStream = null; if (!interviewFiles.isEmpty()) {
//单个文件名称
String interviewFileName = "";
//文件后缀
String fileNameSuffix = "";
//创建一个集合用于 压缩打包的参数
List<Map<String, String>> parms = new ArrayList<>();
//创建一个map集合
Map<String, String> fileMaps =null;
//得到存储文件盘符 例 D:
String root = properties.getValue("upload.root");
//创建一个字节输出流
byteOutPutStream = new ByteArrayOutputStream(); for (InterviewFile file : interviewFiles) { fileMaps = new HashMap<>(); interviewFileName = file.getFileName(); fileNameSuffix = file.getFileSuffix();
//将单个存储路径放入
fileMaps.put("filePath", root.concat(file.getFilePath()));
//将单个文件名放入
fileMaps.put("fileName", interviewFileName.concat("." + fileNameSuffix));
//放入集合
parms.add(fileMaps); }
//压缩文件
FileUtils.batchFileToZIP(parms, byteOutPutStream);
} HttpHeaders headers = new HttpHeaders(); String fileName = null;
try {
fileName = new String("附件.zip".getBytes("UTF-8"), "ISO-8859-1");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); headers.setContentDispositionFormData("attachment", fileName);// 文件名称 ResponseEntity<byte[]> responseEntity = new ResponseEntity<byte[]>(byteOutPutStream.toByteArray(), headers, HttpStatus.CREATED); return responseEntity; }

调用看不懂?那来个简单粗暴的

@RequestMapping(value = "/")
public ResponseEntity<byte[]> downloadInterviewFile() throws Exception { // ---------------------------压缩文件处理-------------------------------
ByteArrayOutputStream byteOutPutStream = new ByteArrayOutputStream(); // 创建一个集合用于 压缩打包的参数
List<Map<String, String>> parms = new ArrayList<>();
// 创建一个map集合
Map<String, String> fileMaps = new HashMap<>();
fileMaps.put("filePath", "D:/文件路径");
fileMaps.put("fileName", "HRM-Package.txt");
Map<String, String> fileMaps1 = new HashMap<>();
fileMaps1.put("filePath", "D:/文件路径1");
fileMaps1.put("fileName", "java.txt");
// 放入集合
parms.add(fileMaps);
parms.add(fileMaps1); // 压缩文件
FileUtils.batchFileToZIP(parms, byteOutPutStream);
// ---------------------------压缩文件处理-------------------------------
HttpHeaders headers = new HttpHeaders(); String fileName = null;
try {
fileName = new String("附件.zip".getBytes("UTF-8"), "ISO-8859-1");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} headers.setContentType(MediaType.APPLICATION_OCTET_STREAM); headers.setContentDispositionFormData("attachment", fileName);// 文件名称 ResponseEntity<byte[]> responseEntity = new ResponseEntity<byte[]>(byteOutPutStream.toByteArray(), headers,
HttpStatus.CREATED); return responseEntity; }

结果:

======================我是分割线======================

这个例子给我了我很大的帮助,因为他是利用ResponseEntity进行下载的,这个下载工具类很好,但是网上一般的例子是基于File的而不是基于byte[]数组的,我的文件是存到数据库中的,文件是用byte[]数组存储的,程序猿一定要深刻理解ZipOutputStream在压缩中的处理关键,话不多说,我直接上我改造的代码工具类

注:FileBank类只有两个属性fileName、fileBlob

public class ZipDownloadUtils {

    /**
* 文件批量压缩
*
* @param parms
*/
public static void batchFileToZIP(List<FileBank> parms, ByteArrayOutputStream byteOutPutStream) { ZipOutputStream zipOut = new ZipOutputStream(byteOutPutStream); try {
for (FileBank value : parms) {
// 使用指定名称创建新的 ZIP 条目 (通俗点就是文件名)
ZipEntry zipEntry = new ZipEntry(value.getFileName());
// 开始写入新的 ZIP 文件条目并将流定位到条目数据的开始处
zipOut.putNextEntry(zipEntry);
  //直接写入到压缩输出流中即可
zipOut.write(value.getFileBlob());
zipOut.closeEntry();
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
zipOut.close();
} catch (IOException e) {
e.printStackTrace();
}
}
} /**
* ResponseEntity下载文件
*
* @param fileName
* @param byteOutPutStream
*/
public static ResponseEntity<byte[]> downloadZIP(String fileName, ByteArrayOutputStream byteOutPutStream) { //下载文件
//String fileName = "批量下载【备案材料】.zip";
HttpHeaders headers = new HttpHeaders();
try {
fileName = new String("附件.zip".getBytes("UTF-8"), "ISO-8859-1");
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
} headers.setContentType(MediaType.APPLICATION_OCTET_STREAM);
headers.setContentDispositionFormData("attachment", fileName);// 文件名称 ResponseEntity<byte[]> responseEntity = new ResponseEntity<byte[]>(byteOutPutStream.toByteArray(), headers, HttpStatus.CREATED);
return responseEntity;
}
}

Controller中代码很简单:

//压缩文件
ByteArrayOutputStream byteOutPutStream = new ByteArrayOutputStream();
ZipDownloadUtils.batchFileToZIP(dataList, byteOutPutStream); //dataList是一个List<FileBank> //下载文件
String fileName = "批量下载【备案材料】.zip";
return ZipDownloadUtils.downloadZIP(fileName, byteOutPutStream);

最后:之所以总结这篇笔记是因为想要用最优雅的代码完成一个文件压缩下载的功能,网上很多用HttpServletResponse原生下载的例子,实在接受不了这么丑陋的代码(可读性奇差,不好维护和修改,通用性很差,代码量太大,等等各种问题不一而足),故而记下此文,即完成了任务又优雅(一个不追求优雅代码的程序员不是好程序员!!!)的实现了,终于可以安心睡觉了……

===============================================================================

如果您觉得此文有帮助,可以打赏点钱给我支付宝或扫描二维码

用Spring中的ResponseEntity文件批量压缩下载的更多相关文章

  1. Spring中MultipartHttpServletRequest实现文件上传

    Spring中MultipartHttpServletRequest实现文件上传 转贴自:http://my.oschina.net/nyniuch/blog/185266 实现图片上传  用户必须能 ...

  2. Java批量压缩下载

    最近做了一些有关批量压缩下载的功能,网上也找了一些资源,但都不是太全面,所以自己整理一份,已备不时之需. 直接上代码: // 获取项目路径 private static String WEBCLASS ...

  3. IE浏览器直接在页面中显示7z文件而不是下载问题解决

    IE浏览器中输入7z文件的完整下载URL后,不是保存文件,而是直接在页面中显示(当然是乱码) 这是因为浏览器对不同的资源文件处理的方式不同,例如图片文件,一般会直接打开,所以我们可以不用7z,使用zi ...

  4. c# 实现文件批量压缩

    今天改一个网站的功能,网站提供一些微信的素材,每个页面对应一套素材,如果会员一张一张下载,那么网站交互性就有点太差了.所以修改的内容就是提供一个按钮,点击按钮将这套图片和网站信息进行打包下载. 思路: ...

  5. Spring中的资源文件框架——Resource

    摘要 Spring4 以后,官方推荐我们使用Java Config来代替applicationContext.xml,声明将Bean交给容器管理. 在Spring Boot中,Java Config的 ...

  6. springMVC实现基本文件夹压缩下载功能

    将文件夹压缩后下载: @Slf4j public class Test { private static final String BASE_PATH = "/root/doc/" ...

  7. Spring中MultipartHttpServletRequest实现文件上传 生成缩略图

    转贴自:http://my.oschina.net/nyniuch/blog/185266 实现图片上传  用户必须能够上传图片,因此需要文件上传的功能.比较常见的文件上传组件有Commons Fil ...

  8. Spring 中使用Properties文件

    Spring提供了加载Properties文件的工具类:org.springframework.beans.factory.config.PropertyPlaceholderConfigurer. ...

  9. Spring中使用属性文件properties的两种方式

    实际项目中,通常将可配置的参数放到属性文件中,例如数据库连接信息.redis连接信息等,便于统一管理.然后通过IoC框架spring将其加载到上下文中,使得程序可以直接使用. 创建mysql.prop ...

随机推荐

  1. 仿有道词典App开发

    最近在学习HCoder提供的仿有道词典App项目,该项目采用MUI为前端框架,服务端采用PHP,底层采用了H5+.

  2. css限制文字显示字数长度,超出部分自动用省略号显示,防止溢出到第二行

    为了保证页面的整洁美观,在很多的时候,我们常需要隐藏超出长度的文字.这在列表条目,题目,名称等地方常用到. 效果如下: 未限制显示长度,如果超出了会溢出到第二行里.严重影响用户体验和显示效果. 我们在 ...

  3. 物联网学习笔记——构建RESTFul平台1

    0.前言     前些时间顺着Yeelink学习了RESTFUL,使用PHP和Slim框架尝试实现简单的REST API,树莓派可通过GET方法获得JSON数据包,通过这种方式实现了树莓派和服务器(我 ...

  4. 【QSBOJ】字符串编辑

    题目链接:https://bbs.csdn.net/topics/390289884?page=1 AC代码: #include<bits/stdc++.h> using namespac ...

  5. swoole之任务和定时器

    一.代码 <?php use Swoole\Server; /** * 面向对象的形式 + task + timer */ class WebSocket { public $server; p ...

  6. java 垒骰子

    垒骰子 赌圣atm晚年迷恋上了垒骰子,就是把骰子一个垒在另一个上边,不能歪歪扭扭,要垒成方柱体. 经过长期观察,atm 发现了稳定骰子的奥秘:有些数字的面贴着会互相排斥! 我们先来规范一下骰子:1 的 ...

  7. vs2010编译C++ 状态标志

    // CTest.cpp : 定义控制台应用程序的入口点. // #include "stdafx.h" #include <iostream> #include &l ...

  8. css - flex 定义排列方向

    flex-direction定义伸缩项目放置在伸缩容器的排列方向,对应有四个值: (1)row:从左到右或从右到左 (2)row-reverse:与row属性相反 (3)column:从上到下排列 ( ...

  9. upper_bound()和low_bound函数的基本使用和理解(转载,已获博主授权)

    版权声明:本文为博主原创文章,未经博主允许不得转载. https://blog.csdn.net/sdz20172133/article/details/80101838 前提:一个非降序列!!!!! ...

  10. Day7 - B - Super A^B mod C FZU - 1759

    Given A,B,C, You should quickly calculate the result of A^B mod C. (1<=A,C<=1000000000,1<=B ...