프로젝트 장면 :
Java는 파일 또는 폴더를 zip으로 압축합니다 (Stream Closed 오류를보고하는 폴더의 여러 파일 문제 수정).
문제 설명:
최근 프로젝트는 다단계 폴더를 zip 파일로 압축해야합니다. 인터넷에서 여러 도구를 발견했고 모두 오류를보고 했으므로이 기사에서 도구 중 하나를 찾으십시오. 개선 한 후 폴더에 여러 파일이 있고 Stream을보고하는 문제가 해결되었습니다. 닫힌 오류 문제입니다.
추신 : Java 초보자,이 변경 사항에 문제가 있는지 모르겠습니다. 위대한 신이 원본 도구 기사에 대한 링크 를 연구하는 데 도움이되기를 바랍니다
.
원인 분석 :
zos.close ();
후퇴가 여러 번 닫힌 후 오류가 다시 실행되는 것을 방지하기 위해 가장 바깥 쪽 try-catch-finally 호출에서 한 번만 닫히도록 변경되었습니다.
원래 논리는 삭제되지 않고 대신 주석 처리됩니다.
해결책:
package utils;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.ArrayList;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipOutputStream;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
// 压缩zip方法
public class ZipUtil {
private static final int BUFFER_SIZE = 2 * 1024;
private static Logger logger = LoggerFactory.getLogger(ZipUtil.class);
/**
* 压缩成ZIP 方法1
*
* @param srcDir
* 压缩文件夹路径
* @param out
* 压缩文件输出流
* @param KeepDirStructure
* 是否保留原来的目录结构,true:保留目录结构;
* false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)
* @throws RuntimeException
* 压缩失败会抛出运行时异常
*/
public static void toZip(String srcDir, OutputStream out, boolean KeepDirStructure){
long start = System.currentTimeMillis();
ZipOutputStream zos = null;
try {
zos = new ZipOutputStream(out);
File sourceFile = new File(srcDir);
compress(sourceFile, zos, sourceFile.getName(), KeepDirStructure);
long end = System.currentTimeMillis();
logger.debug("压缩完成,耗时:" + (end - start) + " ms");
} catch (Exception e) {
e.printStackTrace();
logger.error("系统异常:",e);
} finally {
//zos != null zos.close(); 并不是判断zos是否被关闭 而是这个对象是否创建成功
if (zos != null) {
try {
zos.close();
} catch (IOException e) {
e.printStackTrace();
logger.error("系统异常:",e);
}
}
}
}
/**
* 压缩成ZIP 方法2
*
* @param srcFiles
* 需要压缩的文件列表
* @param out
* 压缩文件输出流
* @throws RuntimeException
* 压缩失败会抛出运行时异常
*/
public static void toZip(List<File> srcFiles, OutputStream out){
long start = System.currentTimeMillis();
ZipOutputStream zos = null;
try {
zos = new ZipOutputStream(out);
for (File srcFile : srcFiles) {
byte[] buf = new byte[BUFFER_SIZE];
zos.putNextEntry(new ZipEntry(srcFile.getName()));
int len;
FileInputStream in = new FileInputStream(srcFile);
while ((len = in.read(buf)) != -1) {
zos.write(buf, 0, len);
}
zos.closeEntry();
in.close();
}
long end = System.currentTimeMillis();
logger.debug("压缩完成,耗时:" + (end - start) + " ms");
} catch (Exception e) {
e.printStackTrace();
logger.error("系统异常:",e);
} finally {
if (zos != null) {
try {
zos.close();
} catch (IOException e) {
e.printStackTrace();
logger.error("系统异常:",e);
}
}
}
}
/**
* 压缩成ZIP 方法2
*
* @param srcFiles
* 需要压缩的文件列表
* @param out
* 压缩文件输出流
* @throws RuntimeException
* 压缩失败会抛出运行时异常
*/
public static void toZip(File[] srcFiles, OutputStream out){
long start = System.currentTimeMillis();
ZipOutputStream zos = null;
try {
zos = new ZipOutputStream(out);
for (File srcFile : srcFiles) {
byte[] buf = new byte[BUFFER_SIZE];
zos.putNextEntry(new ZipEntry(srcFile.getName()));
int len;
FileInputStream in = new FileInputStream(srcFile);
while ((len = in.read(buf)) != -1) {
zos.write(buf, 0, len);
}
zos.closeEntry();
in.close();
}
long end = System.currentTimeMillis();
logger.debug("压缩完成,耗时:" + (end - start) + " ms");
} catch (Exception e) {
e.printStackTrace();
logger.error("系统异常:",e);
} finally {
if (zos != null) {
try {
zos.close();
} catch (IOException e) {
e.printStackTrace();
logger.error("系统异常:",e);
}
}
}
}
/**
*
* 递归压缩方法
*
* @param sourceFile
* 源文件
*
* @param zos
* zip输出流
*
* @param name
* 压缩后的名称
*
* @param KeepDirStructure
* 是否保留原来的目录结构,true:保留目录结构;
*
* false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)
*
* @throws Exception
*
*/
//工具类有逻辑问题 已更改添加注释 便于理解
private static void compress(File sourceFile, ZipOutputStream zos, String name,
boolean KeepDirStructure) throws IOException {
byte[] buf = new byte[BUFFER_SIZE];
//异常改为向上层抛出 避免 zos.close(); 没有执行
// try {
if (sourceFile.isFile()) {
// 向zip输出流中添加一个zip实体,构造器中name为zip实体的文件的名字
zos.putNextEntry(new ZipEntry(name));
// copy文件到zip输出流中
int len;
FileInputStream in = new FileInputStream(sourceFile);
while ((len = in.read(buf)) != -1) {
zos.write(buf, 0, len);
}
// Complete the entry
zos.closeEntry();
in.close();
} else {
File[] listFiles = sourceFile.listFiles();
if (listFiles == null || listFiles.length == 0) {
// 需要保留原来的文件结构时,需要对空文件夹进行处理
if (KeepDirStructure) {
// 空文件夹的处理
zos.putNextEntry(new ZipEntry(name + "/"));
// 没有文件,不需要文件的copy
zos.closeEntry();
}
} else {
for (File file : listFiles) {
// 判断是否需要保留原来的文件结构
if (KeepDirStructure) {
// 注意:file.getName()前面需要带上父文件夹的名字加一斜杠,
// 不然最后压缩包中就不能保留原来的文件结构,即:所有文件都跑到压缩包根目录下了
compress(file, zos, name + "/" + file.getName(), KeepDirStructure);
} else {
compress(file, zos, file.getName(), KeepDirStructure);
}
}
}
}
// }catch(Exception e){
// e.printStackTrace();
// logger.error("系统异常:",e);
// } finally {
// 递归时,不需要把zip输出流关闭,zip输出流的关闭应该是在调用完递归方法后面关闭 这里不再关闭 在最外层关闭
// if (zos != null) {
// try {
// zos.close();
// } catch (IOException e) {
// e.printStackTrace();
// logger.error("系统异常:",e);
// }
//
// }
// }
}
public static void main(String[] args) throws Exception {
/** 测试压缩方法1 */
FileOutputStream fos1 = new FileOutputStream(new File("F:/11.zip"));
ZipUtil.toZip("F:/driverDailyInspection", fos1, true);
/** 测试压缩方法2 */
// List<File> fileList = new ArrayList<>();
// fileList.add(new File("E:/emailTemplate.html"));
// fileList.add(new File("E:/logobottom.jpg"));
// FileOutputStream fos2 = new FileOutputStream(new File("E:/mytest02.zip"));
// ZipUtil.toZip(fileList, fos2);
}
}