java多文件压缩下载的解决方法

Java多文件压缩下载解决方案,供大家参考,具体内容如下

需求:

会员运营平台经过改版后页面增加了许多全部下载链接,上周上线比较仓促,全部下载是一个直接下载ZIP压缩文件的链接,每个ZIP压缩文件都是由公司运营人员将页面需要下载的文件全部压缩成一个ZIP压缩文件,然后通过公司的交易运营平台上传至文件资料系统,会员运营平台则可以直接获取ZIP压缩文件地址进行下载

下面是一个页面示例:

需求分析:

通过上面需求和页面可以分析出,公司运营人员将页面全部需要下载的文件进行ZIP压缩后上传文件资料系统确实是一个紧急的解决方案,但是考虑到后面需求变更,页面需要下载的文件也会跟着变更,每次变更需求,公司运营人员都需要重新进行压缩文件,程序也需要进行相应的修改,这样对于程序的维护性不友好,站在使用系统的客户角度,每次都需要重新上传,因此临时解决方案不再符合软件的良好扩展性和操作方便,因此才有了对页面需要全部下载的文件使用程序压缩处理并下载。

解决思路:

第一步:前端传递Ids字符串

由于会员运营系统显示需要下载的文件是资料系统中的每条文件记录的Id,因此前端页面只需要将需要下载的所有文件Ids字符串(比如:'12,13,14')传递到后台即可.

第二步:后台处理

首先获取到前端传递的ids字符串,将其转换为Integer[]的ids数组,然后调用文件资料微服务根据id列表查询对应的文件记录(包含文件类型和文件地址路径等信息),获取到所有需要下载的文件路径后压缩成ZIP格式的文件进行下载。

具体实现压缩下载方案:

第一种:先压缩成ZIP格式文件,再下载

第二种:边压缩ZIP格式文件边下载(直接输出ZIP流)

前端具体实现代码:

由于全部下载是一个a链接标签,于是使用Ajax异步下载,后来功能实现后点击下载一点反应都没有,一度怀疑是后台出错,但是后台始终没有报错,在网上看了一下Ajax异步不能下载文件(也就是Ajax不支持流类型数据),具体原因可以百度这篇博客,给到的解释是:

原因

ajax的返回值类型是json,text,html,xml类型,或者可以说ajax的接收类型只能是string字符串,不是流类型,所以无法实现文件下载。但用ajax仍然可以获得文件的内容,该文件将被保留在内存中,无法将文件保存到磁盘。这是因为JavaScript无法和磁盘进行交互,否则这会是一个严重的安全问题,js无法调用到浏览器的下载处理机制和程序,会被浏览器阻塞。

解释的还算是比较好的。后面会写一篇=文章详细分析Ajax异步下载解决方案。

接下来考虑使用form表单标签实现,最终配合使用input标签实现了前端传递ids列表的问题,点击a链接标签触发提交form标签即可。

在每一个需要下载的文件增加一个隐藏的input标签,value值是这个文件的id值

具体点击a链接标签提交表单的JS代码:

后端具体实现代码:

第一种方案实现:

第二种方案实现:

附上完整代码:

压缩下载Controller

package com.huajin.jgoms.controller.user;

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.List;
import java.util.Objects;
import java.util.stream.Collectors;

import javax.servlet.http.HttpServletResponse;

import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.lang3.ObjectUtils;
import org.apache.commons.lang3.StringUtils;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;

import com.huajin.baymax.logger.XMsgError;
import com.huajin.baymax.logger.Xlogger;
import com.huajin.common.util.UUIDUtil;
import com.huajin.exchange.domain.sys.FeFileCenter;
import com.huajin.exchange.enums.sys.SysParamKey;
import com.huajin.exchange.po.sys.SysParamPo;
import com.huajin.jgoms.controller.HjBaseController;
import com.huajin.jgoms.service.FeFileCenterService;
import com.huajin.jgoms.service.SysParamService;
import com.huajin.jgoms.util.CompressDownloadUtil;

/**
 * 压缩下载文件
 *
 * @author hongwei.lian
 * @date 2018年9月6日 下午6:29:05
 */
@Controller
@RequestMapping("/compressdownload")
public class CompressDownloadController extends HjBaseController {

 @Autowired
 private FeFileCenterService feFileCenterService;

 @Autowired
 private SysParamService sysParamService;

 /**
 * 多文件压缩下载
 *
 * @author hongwei.lian
 * @date 2018年9月6日 下午6:28:56
 */
 @RequestMapping("/downloadallfiles")
 public void downloadallfiles() {
 //-- 1、根据ids查询下载的文件地址列表
 String ids = request().getParameter("ids");
 if (StringUtils.isEmpty(ids)) {
 return ;
 }
 //-- 将字符串数组改变为整型数组
 Integer[] idsInteger = CompressDownloadUtil.toIntegerArray(ids);
 List<FeFileCenter> fileCenters = feFileCenterService.getFeFileByIds(super.getExchangeId(), idsInteger);
 if (CollectionUtils.isNotEmpty(fileCenters) && ObjectUtils.notEqual(idsInteger.length, fileCenters.size())) {
 //-- 要下载文件Id数组个数和返回的文件地址个数不一致
 return ;
 }

 //-- 2、转换成文件列表
 List<File> files = this.toFileList(fileCenters);
 //-- 检查需要下载多文件列表中文件路径是否都存在
 for (File file : files) {
 if (!file.exists()) {
 //-- 需要下载的文件中存在不存在地址
 return ;
 }
 }

 //-- 3、响应头的设置
 String downloadName = UUIDUtil.getUUID() + ".zip";
 HttpServletResponse response = CompressDownloadUtil.setDownloadResponse(super.response(), downloadName);

 //-- 4、第一种方案:
 //-- 指定ZIP压缩包路径
// String zipFilePath = this.setZipFilePath(downloadName);
// try {
// //-- 将多个文件压缩到指定路径下
// CompressDownloadUtil.compressZip(files, new FileOutputStream(zipFilePath));
// //-- 下载压缩包
// CompressDownloadUtil.downloadFile(response.getOutputStream(), zipFilePath);
// //-- 删除临时生成的ZIP文件
// CompressDownloadUtil.deleteFile(zipFilePath);
// } catch (IOException e) {
// Xlogger.error(XMsgError.buildSimple(CompressDownloadUtil.class.getName(), "downloadallfiles", e));
// }

 //-- 5、第二种方案:
 try {
 //-- 将多个文件压缩写进响应的输出流
 CompressDownloadUtil.compressZip(files, response.getOutputStream());
 } catch (IOException e) {
 Xlogger.error(XMsgError.buildSimple(CompressDownloadUtil.class.getName(), "downloadallfiles", e));
 }

 }

 /**
 * 设置临时生成的ZIP文件路径
 *
 * @param fileName
 * @return
 * @author hongwei.lian
 * @date 2018年9月7日 下午3:54:13
 */
 private String setZipFilePath(String fileName) {
 String zipPath = sysParamService.getCompressDownloadFilePath();
 File zipPathFile = new File(zipPath);
 if (!zipPathFile.exists()) {
 zipPathFile.mkdirs();
 }
 return zipPath + File.separator + fileName;
 }

 /**
 * 将fileCenters列表转换为File列表
 *
 * @param fileCenters
 * @return
 * @author hongwei.lian
 * @date 2018年9月6日 下午6:54:16
 */
 private List<File> toFileList(List<FeFileCenter> fileCenters) {
 return fileCenters.stream()
   .map(feFileCenter -> {
   //-- 获取每个文件的路径
   String filePath = this.getSysFilePath(feFileCenter.getFileTypeId());
   return new File(filePath + feFileCenter.fileLink());})
   .collect(Collectors.toList());
 }

 /**
 * 获取文件类型对应存储路径
 *
 * @param fileTypeId
 * @return
 * @author hongwei.lian
 * @date 2018年9月5日 下午2:01:53
 */
 private String getSysFilePath(Integer fileTypeId){
 SysParamPo sysmParam = sysParamService.getByParamKey(SysParamKey.FC_UPLOAD_ADDRESS.value);
 String filePath = Objects.nonNull(sysmParam) ? sysmParam.getParamValue() : "";
 return filePath + fileTypeId + File.separator;
 }

}

压缩下载工具类

package com.huajin.jgoms.util;

import java.io.BufferedOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;

import javax.servlet.http.HttpServletResponse;

import com.huajin.baymax.logger.XMsgError;
import com.huajin.baymax.logger.Xlogger;

/**
 * 压缩下载工具类
 *
 * @author hongwei.lian
 * @date 2018年9月6日 下午6:34:56
 */
public class CompressDownloadUtil {

 private CompressDownloadUtil() {}

 /**
 * 设置下载响应头
 *
 * @param response
 * @return
 * @author hongwei.lian
 * @date 2018年9月7日 下午3:01:59
 */
 public static HttpServletResponse setDownloadResponse(HttpServletResponse response, String downloadName) {
 response.reset();
 response.setCharacterEncoding("utf-8");
 response.setContentType("application/octet-stream");
 response.setHeader("Content-Disposition", "attachment;fileName*=UTF-8''"+ downloadName);
 return response;
 }

 /**
 * 字符串转换为整型数组
 *
 * @param param
 * @return
 * @author hongwei.lian
 * @date 2018年9月6日 下午6:38:39
 */
 public static Integer[] toIntegerArray(String param) {
 return Arrays.stream(param.split(","))
  .map(Integer::valueOf)
  .toArray(Integer[]::new);
 }

 /**
 * 将多个文件压缩到指定输出流中
 *
 * @param files 需要压缩的文件列表
 * @param outputStream 压缩到指定的输出流
 * @author hongwei.lian
 * @date 2018年9月7日 下午3:11:59
 */
 public static void compressZip(List<File> files, OutputStream outputStream) {
 ZipOutputStream zipOutStream = null;
 try {
 //-- 包装成ZIP格式输出流
 zipOutStream = new ZipOutputStream(new BufferedOutputStream(outputStream));
 // -- 设置压缩方法
 zipOutStream.setMethod(ZipOutputStream.DEFLATED);
 //-- 将多文件循环写入压缩包
 for (int i = 0; i < files.size(); i++) {
 File file = files.get(i);
 FileInputStream filenputStream = new FileInputStream(file);
 byte[] data = new byte[(int) file.length()];
 filenputStream.read(data);
 //-- 添加ZipEntry,并ZipEntry中写入文件流,这里,加上i是防止要下载的文件有重名的导致下载失败
 zipOutStream.putNextEntry(new ZipEntry(i + file.getName()));
 zipOutStream.write(data);
 filenputStream.close();
 zipOutStream.closeEntry();
 }
 } catch (IOException e) {
 Xlogger.error(XMsgError.buildSimple(CompressDownloadUtil.class.getName(), "downloadallfiles", e));
 } finally {
 try {
 if (Objects.nonNull(zipOutStream)) {
 zipOutStream.flush();
 zipOutStream.close();
 }
 if (Objects.nonNull(outputStream)) {
 outputStream.close();
 }
 } catch (IOException e) {
 Xlogger.error(XMsgError.buildSimple(CompressDownloadUtil.class.getName(), "downloadallfiles", e));
 }
 }
 }

 /**
 * 下载文件
 *
 * @param outputStream 下载输出流
 * @param zipFilePath 需要下载文件的路径
 * @author hongwei.lian
 * @date 2018年9月7日 下午3:27:08
 */
 public static void downloadFile(OutputStream outputStream, String zipFilePath) {
 File zipFile = new File(zipFilePath);
 if (!zipFile.exists()) {
 //-- 需要下载压塑包文件不存在
 return ;
 }
 FileInputStream inputStream = null;
 try {
 inputStream = new FileInputStream(zipFile);
 byte[] data = new byte[(int) zipFile.length()];
 inputStream.read(data);
 outputStream.write(data);
 outputStream.flush();
 } catch (IOException e) {
 Xlogger.error(XMsgError.buildSimple(CompressDownloadUtil.class.getName(), "downloadZip", e));
 } finally {
 try {
 if (Objects.nonNull(inputStream)) {
 inputStream.close();
 }
 if (Objects.nonNull(outputStream)) {
 outputStream.close();
 }
 } catch (IOException e) {
 Xlogger.error(XMsgError.buildSimple(CompressDownloadUtil.class.getName(), "downloadZip", e));
 }
 }
 }

 /**
 * 删除指定路径的文件
 *
 * @param filepath
 * @author hongwei.lian
 * @date 2018年9月7日 下午3:44:53
 */
 public static void deleteFile(String filepath) {
 File file = new File(filepath);
 deleteFile(file);
 }

 /**
 * 删除指定文件
 *
 * @param file
 * @author hongwei.lian
 * @date 2018年9月7日 下午3:45:58
 */
 public static void deleteFile(File file) {
 //-- 路径为文件且不为空则进行删除
 if (file.isFile() && file.exists()) {
 file.delete();
 }
 }

}

测试

通过交易运营平台上传测试资料

登录会员运营平台进行下载

下载下来的ZIP格式为文件

解压后,打开文件是否可用:

总结:

这个过程中出现了很多问题,后面会有文章逐步分析出错和解决方案。

上述两种方案都行,但是为了响应速度更快,可以省略压缩成ZIP的临时文件的时间,因此采用了第二种解决方案。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • JavaWeb实现压缩多个文件并下载实例详解

    下面一段代码给大家分享JavaWeb实现压缩多个文件并下载功能,具体代码如下所示: //文件名称 String[] names={"one.jpg","two.jpg","three.jpg","four.jpg"}; //四个文件流 FileInputStream input1 = new FileInputStream(new File("文件路径")); FileInputStream input2

  • java实现文件上传下载和图片压缩代码示例

    分享一个在项目中用的到文件上传下载和对图片的压缩,直接从项目中扒出来的:) 复制代码 代码如下: package com.eabax.plugin.yundada.utils; import java.awt.image.BufferedImage; import java.io.File; import java.io.IOException; import java.text.SimpleDateFormat; import java.util.Date; import java.util.

  • Java 批量文件压缩导出并下载到本地示例代码

    主要用的是org.apache.tools.zip.ZipOutputStream  这个zip流,这里以Execl为例子. 思路首先把zip流写入到http响应输出流中,再把excel的流写入zip流中(这里可以不用生成文件再打包,只需把execl模板读出写好数据输出到zip流中,并为每次的流设置文件名) 例如:在项目webapp下execl文件中 存在1.xls,2.xls,3.xls文件 1.Controller @RequestMapping(value = "/exportAll&qu

  • java压缩文件和下载图片示例

    本文实例为大家分享了java压缩文件和下载图片示例,供大家参考,具体内容如下 主页面index.xml <%@ page language="java" import="java.util.*" pageEncoding="utf-8"%> <html> <head> <title>项目的主页</title> </head> <body> <h2>主页

  • JavaWeb 实现多个文件压缩下载功能

    文件下载时,我们可能需要一次下载多个文件.批量下载文件时,需要将多个文件打包为zip,然后再下载. 实现思路有两种: 一是将所有文件先打包压缩为一个文件,然后下载这个压缩包, 二是一边压缩一边下载,将多个文件逐一写入到压缩文件中.我这里实现了边压缩边下载. 下载样式: 点击下载按钮,会弹出下载框: 下载后就有一个包含刚刚选中的两个文件: 代码实现: FileBean public class FileBean implements Serializable { private Integer f

  • java后台批量下载文件并压缩成zip下载的方法

    本文实例为大家分享了java后台批量下载文件并压缩成zip下载的具体代码,供大家参考,具体内容如下 因项目需要,将服务器上的图片文件压缩打包zip,下载到本地桌面. 首先,前端js: function doQueryPic() { var picsDate = $("#picsDate").val(); var piceDate = $("#piceDate").val(); var picInst = $("#pic_inst").combot

  • java多文件压缩下载的解决方法

    Java多文件压缩下载解决方案,供大家参考,具体内容如下 需求: 会员运营平台经过改版后页面增加了许多全部下载链接,上周上线比较仓促,全部下载是一个直接下载ZIP压缩文件的链接,每个ZIP压缩文件都是由公司运营人员将页面需要下载的文件全部压缩成一个ZIP压缩文件,然后通过公司的交易运营平台上传至文件资料系统,会员运营平台则可以直接获取ZIP压缩文件地址进行下载 下面是一个页面示例: 需求分析: 通过上面需求和页面可以分析出,公司运营人员将页面全部需要下载的文件进行ZIP压缩后上传文件资料系统确实

  • IntelliJ IDEA右键文件夹没有Java Class文件的原因及解决方法

    问题: 在项目里创建文件夹后,发现竟然不能新建class文件,问题详细如下图: 原因分析: 这里涉及到Sources的作用.Sources 一般用于标注类似 src 这种可编译目录.有时候我们项目当中,可能不单单是 src 目录为可编译的,很可能其他一些特别的目录也得是可编译的,因此我们便需要对该目录进行此标注.而在此项目中,只有 Sources 这种可编译目录才可以新建 Java 类和包. 解决方式: (1)选择 File  -> Project Structure  ->  Project

  • Java在指定路径上创建文件提示不存在解决方法

    如果 d:\upload\file\ 文件夹不存在,会报错 String strPath = "d:\\upload\\file\\2.mp3"; File file = new File(strPath); if(!file.exists())){ file.createNewFile(); } 以下会创建文件夹 d:\\upload\\file\\2.mp3\ String strPath = "d:\\upload\\file\\2.mp3"; File fi

  • Java 内存溢出的原因和解决方法

    你是否遇到过Java应用程序卡顿或突然崩溃的情况?您可能遇到过Java内存泄漏.在本文中,我们将深入研究Java内存泄漏的确切原因,并推荐一些最好的工具来防止内存泄漏发生. 什么是JAVA内存泄漏? 简单地说,Java内存泄漏是指对象不再被应用程序使用,而是在工作内存中处于活动状态. 在Java和大多数其他编程语言中,垃圾收集器的任务是删除不再被应用程序引用的对象.如果不选中,这些对象将继续消耗系统内存,并最终导致崩溃.有时java内存泄漏崩溃不会输出错误,但通常错误会以java.lang.Ou

  • 教你用MAT工具分析Java堆内存泄漏问题的解决方法

    一.MAT概述与安装 MAT,全称Memory Analysis Tools,是一款分析Java堆内存的工具,可以快速定位到堆内泄漏问题.该工具提供了两种使用方式,一种是插件版,可以安装到Eclipse使用,另一种是独立版,可以直接解压使用. 我把独立版MAT安装包放到了网盘上,方便直接下载 链接: https://pan.baidu.com/s/1DVHlHuSfi_4TVl2ei5YuLA 提取码: 42qt 独立版解压后,其内部文件是这样的-- 这里有一个MemoryAnalyzer.in

  • Java创建文件且写入内容的方法

    前两天在项目中因为要通过http请求获取一个比较大的json数据(300KB左右)并且保存,思来想去,最后还是决定将获取到的json数据以文件的形式保存下来,每次使用的时候去读取文件就可以了. 废话不多说了,直接上代码. 以下是代码截图,文章结尾会有完成的代码文件可供下载. 创建文件方法: 写入文件内容方法: 删除文件方法: 测试: 关于文件创建,写入内容,删除.可以根据自己的情况再稍作修改. 以下是代码类. package com.file.run; import java.io.Buffer

  • Android实现zip文件压缩及解压缩的方法

    本文实例讲述了Android实现zip文件压缩及解压缩的方法.分享给大家供大家参考.具体如下: DirTraversal.java如下: package com.once; import java.io.File; import java.util.ArrayList; import java.util.LinkedList; /** * 文件夹遍历 * @author once * */ public class DirTraversal { //no recursion public sta

  • 基于vue-cli npm run build之后vendor.js文件过大的解决方法

    问题 vue-cli npm run build命令默认把dependencies中的依赖统一打包,导致vendor.js文件过大,出现首屏加载过于缓慢的问题. 解决方案 像vue.axios.element-ui这些基本上不会改变的依赖我们可以把它们用cdn导入,没有必要打包到vendor.js中. 1.在项目根目录index.html使用cdn节点导入 <div id="app"></div> <!-- 先引入 Vue --> <!--开发

随机推荐