Java实现解压zip压缩包的两种方法(支持多层级)
前言
当我们需要上传大量文件时,压缩文件可以大大减少传输时间和网络带宽,在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(); } } } }
到此这篇关于Java实现解压zip压缩包的两种方法(支持多层级)的文章就介绍到这了,更多相关Java 解压zip压缩包内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
springboot引用kettle实现对接oracle数据的示例代码
这篇文章主要介绍了springboot引用kettle实现对接oracle数据,其实kettle集成到springboot里面没有多少代码,这个功能最主要的还是ktr文件的编写,只要ktr编写好了,放到指定文件夹下,写个定时任务就完事了,需要的朋友可以参考下2022-12-12
最新评论