Java实现解压zip压缩包(支持多层级)

本文介绍了在Java中使用ZipInputStream解压文件时遇到的中文文件名问题,推荐使用ApacheCommonsCompress库的ZipArchiveInputStream解决跨平台编码问题。同时,作者还展示了如何将这些操作整理为工具类,便于在实际项目中应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

🍓 简介:java系列技术分享(👉持续更新中…🔥)
🍓 初衷:一起学习、一起进步、坚持不懈
🍓 如果文章内容有误与您的想法不一致,欢迎大家在评论区指正🙏
🍓 希望这篇文章对你有所帮助,欢迎点赞 👍 收藏 ⭐留言 📝

🍓 更多文章请点击
在这里插入图片描述在这里插入图片描述

1

前言

当我们需要上传大量文件时,压缩文件可以大大减少传输时间和网络带宽,在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格式,造成这种现象的原因有几种可能:

  1. 压缩包中包含中文
  2. 在linux上压缩,在windows上使用解压工具解压
  3. 使用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();
                }
            }
        }
    }

在这里插入图片描述在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Dream_sky分享

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值