项目经历:
在项目中遇到了一个需求就是用户批量下载文件,方案有两种,第一种,就是前端去调用多个下载接口实现,第二种,就是服务端把文件打成zip包去导出文件。
一 . 下载单个文件
File file = new File("文件地址(/var/1.doc)");
// 以流的形式下载文件。
InputStream fis = new BufferedInputStream(new FileInputStream(file.getPath()));
byte[] buffer = new byte[fis.available()];
fis.read(buffer);
fis.close();
// 清空response
response.reset();
OutputStream toClient = new BufferedOutputStream(response.getOutputStream());
response.setContentType("application/octet-stream");
response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode("文件名称(1.doc)","UTF-8"));
toClient.write(buffer);
toClient.flush();
toClient.close();
二 . 压缩多个文件成一个zip下载Controller请求方法
/**
* @Description 下载文件单个源文件,多个zip文件
*@Param [session, request, response, data]
*@Return void
*@Author Wzq
*@Date 2019/11/25
*@Time 11:15
*/
@RequestMapping(value = "/downMeetingMaterialFile.action")
public void downMeetingMaterialFile(final HttpSession session,
final HttpServletRequest request,
final HttpServletResponse response,
@RequestParam(value = "Data") String data) throws Exception {
List<File> fileList = Lists.newArrayList();
File file1 = new File("1.doc");
File file2 = new File("2.doc");
fileList.add(file1);
fileList.add(file2);
//随机名称临时存放zip的目录,导出后删除
String destDir = "/var/"+ File.separator + UUID.randomUUID().toString();
File destDirFile = new File(destDir);
if(!destDirFile.exists()){
destDirFile.mkdirs();
}
try {
BatchDownFilesUtils.downLoadFiles(fileList,zipPath+ ".zip",request,response);
} finally {
FileUtils.delFile(destDirFile);
}
}
下面是我FileUtils工具类:
package com.meeno.trainsys.util;
import java.io.File;
/**
* @description: 文件工具类
* @author: Wzq
* @create: 2019-11-25 14:57
*/
public class FileUtils {
public static boolean delFile(File file) {
if (!file.exists()) {
return false;
}
if (file.isFile()) {
return file.delete();
} else {
File[] files = file.listFiles();
for (File f : files) {
delFile(f);
}
return file.delete();
}
}
}
下面是zip批量导出工具类:
package com.meeno.trainsys.util;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.*;
import java.net.URLEncoder;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;
/**
* @description: 批量下载文件
* @author: Wzq
* @create: 2019-11-25 10:17
*/
public class BatchDownFilesUtils {
/**
*@Description 输入文件集合,zipPath临时路径,request,response导出zip文件
*@Param [files, zipPath, request, response]
*@Return javax.servlet.http.HttpServletResponse
*@Author Wzq
*@Date 2019/11/25
*@Time 19:49
*/
public static HttpServletResponse downLoadFiles(List<File> files,
String zipPath,
HttpServletRequest request, HttpServletResponse response)
throws Exception {
try {
/**这个集合就是你想要打包的所有文件,
* 这里假设已经准备好了所要打包的文件
*/
//List<File> files = new ArrayList<File>();
/**创建一个临时压缩文件,
* 我们会把文件流全部注入到这个文件中
* 这里的文件你可以自定义是.rar还是.zip
* 这里的file路径发布到生产环境时可以改为
*/
File file = new File(zipPath);
if (!file.exists()){
file.createNewFile();
}
response.reset();
//response.getWriter()
//创建文件输出流
FileOutputStream fous = new FileOutputStream(file);
/**打包的方法我们会用到ZipOutputStream这样一个输出流,
* 所以这里我们把输出流转换一下*/
// org.apache.tools.zip.ZipOutputStream zipOut
// = new org.apache.tools.zip.ZipOutputStream(fous);
ZipOutputStream zipOut
= new ZipOutputStream(fous);
/**这个方法接受的就是一个所要打包文件的集合,
* 还有一个ZipOutputStream
*/
zipFile(files, zipOut);
zipOut.close();
fous.close();
return downloadZip(file,response);
}catch (Exception e) {
e.printStackTrace();
}
/**直到文件的打包已经成功了,
* 文件的打包过程被我封装在FileUtil.zipFile这个静态方法中,
* 稍后会呈现出来,接下来的就是往客户端写数据了
*/
// OutputStream out = response.getOutputStream();
return response ;
}
/**
* 把接受的全部文件打成压缩包
* @param List<File>;
* @param org.apache.tools.zip.ZipOutputStream
*/
private static void zipFile
(List files,ZipOutputStream outputStream) {
int size = files.size();
for(int i = 0; i < size; i++) {
File file = (File) files.get(i);
zipFile(file, outputStream);
}
}
/**
*@Description 下载zip文件,删除缓存文件
*@Param [file, response]
*@Return javax.servlet.http.HttpServletResponse
*@Author Wzq
*@Date 2019/11/25
*@Time 19:48
*/
private static HttpServletResponse downloadZip(File file, HttpServletResponse response) {
try {
// 以流的形式下载文件。
InputStream fis = new BufferedInputStream(new FileInputStream(file.getPath()));
byte[] buffer = new byte[fis.available()];
fis.read(buffer);
fis.close();
// 清空response
response.reset();
OutputStream toClient = new BufferedOutputStream(response.getOutputStream());
response.setContentType("application/octet-stream");
response.setHeader("Content-Disposition", "attachment;filename=" + URLEncoder.encode(file.getName(),"UTF-8"));
toClient.write(buffer);
toClient.flush();
toClient.close();
} catch (IOException ex) {
ex.printStackTrace();
}finally{
try {
File f = new File(file.getPath());
f.delete();
} catch (Exception e) {
e.printStackTrace();
}
}
return response;
}
/**
* 根据输入的文件与输出流对文件进行打包
* @param File
* @param org.apache.tools.zip.ZipOutputStream
*/
private static void zipFile(File inputFile,
ZipOutputStream ouputStream) {
try {
if(inputFile.exists()) {
/**如果是目录的话这里是不采取操作的,
* 至于目录的打包正在研究中
*/
if (inputFile.isFile()) {
FileInputStream IN = new FileInputStream(inputFile);
BufferedInputStream bins = new BufferedInputStream(IN, 512);
//org.apache.tools.zip.ZipEntry
ZipEntry entry = new ZipEntry(inputFile.getName());
ouputStream.putNextEntry(entry);
// 向压缩文件中输出数据
int nNumber;
byte[] buffer = new byte[512];
while ((nNumber = bins.read(buffer)) != -1) {
ouputStream.write(buffer, 0, nNumber);
}
// 关闭创建的流对象
bins.close();
IN.close();
} else {
try {
File[] files = inputFile.listFiles();
for (int i = 0; i < files.length; i++) {
zipFile(files[i], ouputStream);
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
} catch (Exception e) {
e.printStackTrace();
}
}
}