Skip to content

引言

在项目中,可以通过标准库 java.util.zip 来实现ZIP文件的压缩和解压缩。压缩时,将多个文件或目录打包成一个 ZIP 文件;解压缩时,将 ZIP 文件中的内容提取到指定目录。

添加依赖

项目默认已经包含了 java.util.zip ,因此无需额外添加依赖。

压缩、解压缩工具

xiaomayi-common/xiaomayi-core 核心模块的 utils 工具目录中创建并定义了文件压缩、解压缩工具类。

js
package com.xiaomayi.core.utils;

import java.io.*;
import java.nio.charset.Charset;
import java.util.Enumeration;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipFile;
import java.util.zip.ZipOutputStream;

/**
 * <p>
 * ZIP压缩工具类
 * </p>
 * 特别注意:依赖非静态set方法,切记不然无法读取值
 *
 * @author 小蚂蚁云团队
 * @since 2024-02-21
 */
public class ZipUtils {

    private static final int BUFFER_SIZE = 20 * 1024;

    /**
     * 压缩文件夹到指定zip文件
     *
     * @param srcDir     源文件夹
     * @param targetFile 目标知道zip文件
     * @throws IOException IO异常,抛出给调用者处理
     */
    public static void zip(String srcDir, String targetFile, boolean KeepDirStructure) throws IOException {
        try (
                OutputStream outputStream = new FileOutputStream(targetFile);
        ) {
            zip(srcDir, outputStream, KeepDirStructure);
        }
    }

    /**
     * 压缩成ZIP
     *
     * @param srcDir           压缩文件夹路径
     * @param out              压缩文件输出流
     * @param KeepDirStructure 是否保留原来的目录结构,true:保留目录结构;
     *                         false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)
     * @throws RuntimeException 压缩失败会抛出运行时异常
     */
    public static void zip(String srcDir, OutputStream out, boolean KeepDirStructure) throws RuntimeException {
        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();
            System.out.println("压缩完成,耗时:" + (end - start) + " ms");
        } catch (Exception e) {
            throw new RuntimeException("zip error from ZipUtils", e);
        } finally {
            if (zos != null) {
                try {
                    zos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    /**
     * 压缩文件
     *
     * @param srcFiles   需要压缩的文件列表
     * @param targetFile 压缩文件输出
     * @throws IOException 压缩失败会抛出运行时异常
     */
    public static void zip(List<File> srcFiles, String targetFile) throws IOException {
        try (
                OutputStream outputStream = new FileOutputStream(targetFile);
        ) {
            zip(srcFiles, outputStream);
        }
    }

    /**
     * 文件压缩
     *
     * @param srcFiles 需要压缩的文件列表
     * @param out      压缩文件输出流
     * @throws RuntimeException 压缩失败会抛出运行时异常
     */
    public static void zip(List<File> srcFiles, OutputStream out) throws RuntimeException {
        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();
            System.out.println("压缩完成,耗时:" + (end - start) + " ms");
        } catch (Exception e) {
            throw new RuntimeException("zip error from ZipUtils", e);
        } finally {
            if (zos != null) {
                try {
                    zos.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    /**
     * 递归压缩方法
     *
     * @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 Exception {
        byte[] buf = new byte[BUFFER_SIZE];
        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);
                    }
                }
            }
        }
    }

    /**
     * 文件解压
     *
     * @param zipFileName 解压文件路径
     * @param destDirPath 解压目标文件路径
     */
    public static void unzip(String zipFileName, String destDirPath) {
        File srcFile = new File(zipFileName);
        unZip(srcFile, destDirPath);
    }

    /**
     * 文件解压
     *
     * @param srcFile     解压文件路径
     * @param destDirPath 解压目标文件路径
     * @throws RuntimeException
     */
    public static void unZip(File srcFile, String destDirPath) throws RuntimeException {
        long start = System.currentTimeMillis();
        // 判断源文件是否存在
        if (!srcFile.exists()) {
            throw new RuntimeException(srcFile.getPath() + "所指文件不存在");
        }
        // 开始解压
        ZipFile zipFile = null;
        try {
            //指定编码格式,不然会报错:java.lang.IllegalArgumentException: MALFORMED
            zipFile = new ZipFile(srcFile, Charset.forName("GBK"));
            Enumeration<?> entries = zipFile.entries();
            while (entries.hasMoreElements()) {
                ZipEntry entry = (ZipEntry) entries.nextElement();
                System.out.println("解压" + entry.getName());
                // 如果是文件夹,就创建个文件夹
                if (entry.isDirectory()) {
                    String dirPath = destDirPath + "/" + entry.getName();
                    File dir = new File(dirPath);
                    dir.mkdirs();
                } else {
                    // 如果是文件,就先创建一个文件,然后用io流把内容copy过去
                    File targetFile = new File(destDirPath + "/" + entry.getName());
                    // 保证这个文件的父文件夹必须要存在
                    if (!targetFile.getParentFile().exists()) {
                        targetFile.getParentFile().mkdirs();
                    }
                    targetFile.createNewFile();
                    // 将压缩文件内容写入到这个文件中
                    InputStream is = zipFile.getInputStream(entry);
                    FileOutputStream fos = new FileOutputStream(targetFile);
                    int len;
                    byte[] buf = new byte[BUFFER_SIZE];
                    while ((len = is.read(buf)) != -1) {
                        fos.write(buf, 0, len);
                    }
                    // 关流顺序,先打开的后关闭
                    fos.close();
                    is.close();
                }
            }
            long end = System.currentTimeMillis();
            System.out.println("解压完成,耗时:" + (end - start) + " ms");
        } catch (Exception e) {
            throw new RuntimeException("unzip error from ZipUtils", e);
        } finally {
            if (zipFile != null) {
                try {
                    zipFile.close();
                } catch (IOException e) {
                    e.printStackTrace();
                }
            }
        }
    }

}

压缩、解压缩实现

E盘 准备三个文件,分别是 文件1.xlsx文件2.xlsx文件1.docx,如下图:

文件压缩、解压缩实现:

js
package com.xiaomayi.admin.controller.demo;

import com.xiaomayi.core.utils.R;
import com.xiaomayi.core.utils.ZipUtils;
import lombok.AllArgsConstructor;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

/**
 * <p>
 * 压缩、解压缩文件 前端控制器
 * </p>
 *
 * @author 小蚂蚁云团队
 * @since 2024-05-26
 */
@Slf4j
@RestController
@RequestMapping("/zip")
@AllArgsConstructor
public class UnZipController {

    /**
     * 文件压缩
     *
     * @return 返回结果
     * @throws IOException 异常处理
     */
    @GetMapping("/compress")
    public R compress() throws IOException {
        // 设置源文件路径
        String sourceFolderPath = "E:\\compress";
        // 设置压缩文件路径
        String zipFilePath = "E:\\compress.zip";
        // 实现文件压缩
        ZipUtils.zip(sourceFolderPath, zipFilePath, true);
        return R.ok();
    }

    /**
     * 指定文件列表压缩
     *
     * @return 返回结果
     * @throws IOException 异常处理
     */
    @GetMapping("/compress2")
    public R compress2() throws IOException {
        // 设置文件列表
        List<File> srcFiles = new ArrayList<>();
        srcFiles.add(new File("E:\\compress\\文件1.xlsx"));
        srcFiles.add(new File("E:\\compress\\文件2.xlsx"));
        srcFiles.add(new File("E:\\compress\\文件1.docx"));
        // 压缩文件路径
        String zipFilePath = "E:\\compress.zip";
        // 实现文件压缩
        ZipUtils.zip(srcFiles, zipFilePath);
        return R.ok();
    }

    /**
     * 文件解压缩
     *
     * @return 返回结果
     * @throws IOException 异常处理
     */
    @GetMapping("/decompress")
    public R decompress() throws IOException {
        // 设置压缩文件路径
        String zipFilePath = "E:\\compress.zip";
        // 设置解压文件路径
        String destDirectory = "E:\\compress2";
        // 实现文件解压缩
        ZipUtils.unzip(zipFilePath, destDirectory);
        return R.ok();
    }

}

总结

在项目中,可以通过 OutputStreamjava.util.zip 包实现ZIP文件的压缩和解压缩。压缩时,将文件或目录写入 ZipOutputStream;解压缩时,从 ZipInputStream 读取文件并写入目标目录。

小蚂蚁云团队 · 提供技术支持

小蚂蚁云 新品首发
新品首发,限时特惠,抢购从速! 全场95折
赋能开发者,助理企业发展,提供全方位数据中台解决方案。
获取官方授权