🍓 简介:java系列技术分享(👉持续更新中…🔥)
🍓 初衷:一起学习、一起进步、坚持不懈
🍓 如果文章内容有误与您的想法不一致,欢迎大家在评论区指正🙏
🍓 希望这篇文章对你有所帮助,欢迎点赞 👍 收藏 ⭐留言 📝
文章目录
前言
当我们需要上传大量文件时,压缩文件可以大大减少传输时间和网络带宽,在Java中提供了压缩和解压缩文件的功能,可以使用
java.util.zip
包中的类来实现。但是也有很多不足
, 本篇将对如何使用 Java 实现解压缩进行简单总结。
ZIP
:最常见的压缩文件格式之一,可以存储一个或多个文件,并可在不同的操作系统中进行解压缩。
一、使用ZipInputStream解压文件流(不支持中文)
public static List<File> unzipInputStream(InputStream zipInputStream) {
List<File> fileList = new ArrayList<>();
try (ZipInputStream zip = new ZipInputStream(zipInputStream)) {
ZipEntry zipEntry = null;
while ((zipEntry = zip.getNextEntry()) != null) {
String fileName_zip = zipEntry.getName();
File file = new File(fileName_zip);
if (fileName_zip.endsWith("/")) {
file.mkdir();
continue;
} else {
BufferedOutputStream outputStream = new BufferedOutputStream(new FileOutputStream(file));
byte[] byte_s = new byte[1024];
int num;
while ((num = zip.read(byte_s, 0, byte_s.length)) > 0) {
outputStream.write(byte_s, 0, num);
}
outputStream.close();
}
fileList.add(file);
}
} catch (Exception e) {
e.printStackTrace();
}
return fileList;
}
在调用JDK自带的zipfile读取压缩包文件的时候,出现了以下错误: MALFORMED
java.lang.IllegalArgumentException: MALFORMED
at java.util.zip.ZipCoder.toString(ZipCoder.java:58)
at java.util.zip.ZipInputStream.readLOC(ZipInputStream.java:300)
at java.util.zip.ZipInputStream.getNextEntry(ZipInputStream.java:122)
at com.wonder.meta.util.ZipUtils.unzipInputStream(ZipUtils.java:26)
1.1 原因描述
后来经过检查,发现压缩包内有一个文件的名字带有中文,解析压缩包中,读取中文文件导致报错
由于操作系统平台的差异,导致zip压缩包的编码格式不同,windows默认使用GB2312格式,mac和linux默认使用utf-8格式,造成这种现象的原因有几种可能:
- 压缩包中包含中文
- 在linux上压缩,在windows上使用解压工具解压
- 使用jdk api直接解压
1.2 解决方案
推荐使用(经过测试)Apache commons-compress 工具包解决跨平台编码问题
二、使用ZipArchiveInputStream解压(推荐)
2.1 引入pom依赖
<dependency>
<groupId>org.apache.commons</groupId>
<artifactId>commons-compress</artifactId>
<version>1.21</version>
</dependency>
ZipInputStream 和 ZipArchiveInputStream 都是 Java 中用于处理 Zip 文件的类,但是它们之间有一些区别。
-
ZipInputStream 是 Java 标准库中提供的类,而 ZipArchiveInputStream 是 Apache Commons Compress 库中提供的类。
-
ZipInputStream 只能读取普通的 Zip 文件,而 ZipArchiveInputStream 支持读取多种压缩格式,包括 Zip、Gzip、Tar、Jar 等。
-
ZipArchiveInputStream 提供了更多的选项和功能,例如可以设置编码方式、支持密码保护的 Zip 文件等。
-
ZipArchiveInputStream 的性能比 ZipInputStream 更好,在处理大型 Zip 文件时表现更出色。
2.2 进行改造后
public static List<File> unzipInputStream(InputStream zipInputStream) {
List<File> fileList = new ArrayList<>();
try (ZipArchiveInputStream zip = new ZipArchiveInputStream(zipInputStream)) {
ZipArchiveEntry zipEntry = null;
while ((zipEntry = zip.getNextZipEntry()) != null) {
String fileName_zip = zipEntry.getName();
File file = new File(fileName_zip);
if (fileName_zip.endsWith("/")) {
file.mkdir();
continue;
} else {
BufferedOutputStream outputStream = new BufferedOutputStream(new FileOutputStream(file));
byte[] byte_s = new byte[1024];
int num;
while ((num = zip.read(byte_s, 0, byte_s.length)) > 0) {
outputStream.write(byte_s, 0, num);
}
outputStream.close();
}
fileList.add(file);
}
} catch (Exception e) {
e.printStackTrace();
}
return fileList;
}
测试正常
三、整理为工具类
- 包含获取目录下的图片…(包括子目录)
- 定义文件过滤器(过滤想留下的文件)
3.1 ZipUtils
public class ZipUtils {
public static List<File> unzipInputStream(InputStream zipInputStream) {
List<File> fileList = new ArrayList<>();
try (ZipArchiveInputStream zip = new ZipArchiveInputStream(zipInputStream)) {
ZipArchiveEntry zipEntry = null;
while ((zipEntry = zip.getNextZipEntry()) != null) {
String fileName_zip = zipEntry.getName();
File file = new File(fileName_zip);
if (fileName_zip.endsWith("/")) {
file.mkdir();
continue;
} else {
BufferedOutputStream outputStream = new BufferedOutputStream(new FileOutputStream(file));
byte[] byte_s = new byte[1024];
int num;
while ((num = zip.read(byte_s, 0, byte_s.length)) > 0) {
outputStream.write(byte_s, 0, num);
}
outputStream.close();
}
fileList.add(file);
}
} catch (Exception e) {
e.printStackTrace();
}
return fileList;
}
// 单独列出方法获取目录下的图片
public static List<File> listAll(File parent) {
List<File> fileList = new ArrayList<>();
MyFilter myFilter = new MyFilter();
File[] children = parent.listFiles(myFilter);
if(children != null){
for (int i = 0; i < children.length; i++) {
// 如果子文件是个文件夹,则递归调用
if (!children[i].isFile()) {
listAll(children[i]);
} else {
fileList.add(children[i]);
}
}
}
return fileList;
}
//定义文件过滤器
static class MyFilter implements FilenameFilter {
@Override
public boolean accept(File dir, String name) {
return name.contains(".png") || dir.isDirectory();
}
}
}
3.2 这里列出实现类的方法作为参考
记得删除临时文件
@Override
public void batchUploadSign(MultipartFile multipartFile) {
List<File> fileList = null;
try {
fileList = ZipUtils.unzipInputStream(multipartFile.getInputStream());
//fileList可能存在目录
for (File fileTemp : fileList) {
List<File> fileTempList = ZipUtils.listAll(fileTemp);
if (CollectionUtil.isNotEmpty(fileTempList)) {
fileList.addAll(fileTempList);
}
}
if (CollectionUtil.isNotEmpty(fileList)) {
log.info("省略业务逻辑");
}
} catch (IOException e) {
throw new RuntimeException(e);
} finally {
//删除临时文件
if (null != fileList) {
for (File file1 : fileList) {
file1.delete();
}
}
}
}