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

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

🍓 简介: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();
                }
            }
        }
    }

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

  • 30
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
要在不解压缩的情况下替换zip压缩包中的指定文件,可以使用JavaZipFile和ZipOutputStream类。 首先,使用ZipFile读取原始Zip文件,并且使用ZipOutputStream创建一个新的Zip文件。然后,遍历ZipFile中的所有条目,并将所有条目写入新的ZipOutputStream中,除了要替换的文件。当到达要替换的文件时,使用ZipEntry将文件添加到新的ZipOutputStream中,以便替换原始文件。 以下是一个示例代码: ```java import java.io.*; import java.util.*; import java.util.zip.*; public class ReplaceZipFileEntry { public static void main(String[] args) throws Exception { String zipFileName = "example.zip"; String fileToReplace = "file.txt"; String replacementFileName = "replacement.txt"; // Open the original zip file and create a new one File zipFile = new File(zipFileName); File tempZipFile = new File("temp.zip"); ZipFile zip = new ZipFile(zipFile); ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(tempZipFile)); // Loop through all the entries in the original zip file Enumeration entries = zip.entries(); while (entries.hasMoreElements()) { ZipEntry entry = (ZipEntry) entries.nextElement(); // If this is the file to be replaced, add the replacement file instead if (entry.getName().equals(fileToReplace)) { File replacementFile = new File(replacementFileName); FileInputStream fis = new FileInputStream(replacementFile); zos.putNextEntry(new ZipEntry(entry.getName())); byte[] buffer = new byte[1024]; int length; while ((length = fis.read(buffer)) > 0) { zos.write(buffer, 0, length); } fis.close(); } else { // Otherwise, copy the original entry to the new zip file zos.putNextEntry(entry); InputStream is = zip.getInputStream(entry); byte[] buffer = new byte[1024]; int length; while ((length = is.read(buffer)) > 0) { zos.write(buffer, 0, length); } is.close(); } } // Close the streams and delete the original zip file zos.close(); zip.close(); zipFile.delete(); // Rename the temporary zip file to the original name tempZipFile.renameTo(zipFile); } } ``` 在此示例中,我们将替换example.zip文件中名为file.txt的文件。我们使用replacement.txt文件替换它。在代码中,我们首先打开原始zip文件并创建一个新的zip文件(temp.zip)。然后,我们遍历原始zip文件中的所有条目,并将它们添加到新的zip文件中。如果我们到达要替换的文件,我们使用ZipEntry添加replacement.txt文件。最后,我们关闭所有流,删除原始zip文件,并将临时zip文件重命名为原始zip文件的名称。 请注意,这种方法适用于仅替换一个或两个文件的情况。如果您需要替换多个文件,最好解压缩整个zip文件,替换所需的文件,然后重新压缩整个zip文件。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Dream_sky分享

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

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

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

打赏作者

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

抵扣说明:

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

余额充值