使用java API進行zip遞迴壓縮資料夾以及解壓

字母哥部落格發表於2020-08-10

一、概述

在本篇文章中,給大家介紹一下如何將檔案進行zip壓縮以及如何對zip包解壓。所有這些都是使用Java提供的核心庫java.util.zip來實現的。

二、壓縮檔案

首先我們來學習一個簡單的例子-壓縮單個檔案。將一個名為test1.txt的檔案壓縮到一個名為Compressed.zip的zip檔案中。

public class ZipFile {
    public static void main(String[] args) throws IOException {
       
        //輸出壓縮包
        FileOutputStream fos = new FileOutputStream("src/main/resources/compressed.zip");
        ZipOutputStream zipOut = new ZipOutputStream(fos);

        //被壓縮檔案
        File fileToZip = new File("src/main/resources/test1.txt");
        FileInputStream fis = new FileInputStream(fileToZip);
        
        //向壓縮包中新增檔案
        ZipEntry zipEntry = new ZipEntry(fileToZip.getName());
        zipOut.putNextEntry(zipEntry);
        byte[] bytes = new byte[1024];
        int length;
        while((length = fis.read(bytes)) >= 0) {
            zipOut.write(bytes, 0, length);
        }
        zipOut.close();
        fis.close();
        fos.close();
    }
}

三、壓縮多個檔案

接下來,我們看看如何將多個檔案壓縮為一個zip檔案。我們將把test1.txttest2.txt壓縮成multiCompressed.zip

public class ZipMultipleFiles {
    public static void main(String[] args) throws IOException {
        List<String> srcFiles = Arrays.asList("src/main/resources/test1.txt", "src/main/resources/test2.txt");
        FileOutputStream fos = new FileOutputStream("src/main/resources/multiCompressed.zip");
        ZipOutputStream zipOut = new ZipOutputStream(fos);
        //向壓縮包中新增多個檔案
        for (String srcFile : srcFiles) {
            File fileToZip = new File(srcFile);
            FileInputStream fis = new FileInputStream(fileToZip);
            ZipEntry zipEntry = new ZipEntry(fileToZip.getName());
            zipOut.putNextEntry(zipEntry);
 
            byte[] bytes = new byte[1024];
            int length;
            while((length = fis.read(bytes)) >= 0) {
                zipOut.write(bytes, 0, length);
            }
            fis.close();
        }
        zipOut.close();
        fos.close();
    }
}

四、壓縮目錄

下面的例子,我們將zipTest目錄及該目錄下的遞迴子目錄檔案,全都壓縮到dirCompressed.zip中

public class ZipDirectory {
    public static void main(String[] args) throws IOException, FileNotFoundException {
        //被壓縮的資料夾
        String sourceFile = "src/main/resources/zipTest"; 
        //壓縮結果輸出,即壓縮包
        FileOutputStream fos = new FileOutputStream("src/main/resources/dirCompressed.zip");
        ZipOutputStream zipOut = new ZipOutputStream(fos);
        File fileToZip = new File(sourceFile);
        //遞迴壓縮資料夾
        zipFile(fileToZip, fileToZip.getName(), zipOut);
        //關閉輸出流
        zipOut.close();
        fos.close();
    }
     

    /**
     * 將fileToZip資料夾及其子目錄檔案遞迴壓縮到zip檔案中
     * @param fileToZip 遞迴當前處理物件,可能是資料夾,也可能是檔案
     * @param fileName fileToZip檔案或資料夾名稱
     * @param zipOut 壓縮檔案輸出流
     * @throws IOException
     */
    private static void zipFile(File fileToZip, String fileName, ZipOutputStream zipOut) throws IOException {
        //不壓縮隱藏資料夾
        if (fileToZip.isHidden()) {
            return;
        }
        //判斷壓縮物件如果是一個資料夾
        if (fileToZip.isDirectory()) {
            if (fileName.endsWith("/")) {
                //如果資料夾是以“/”結尾,將資料夾作為壓縮箱放入zipOut壓縮輸出流
                zipOut.putNextEntry(new ZipEntry(fileName));
                zipOut.closeEntry();
            } else {
                //如果資料夾不是以“/”結尾,將資料夾結尾加上“/”之後作為壓縮箱放入zipOut壓縮輸出流
                zipOut.putNextEntry(new ZipEntry(fileName + "/"));
                zipOut.closeEntry();
            }
            //遍歷資料夾子目錄,進行遞迴的zipFile
            File[] children = fileToZip.listFiles();
            for (File childFile : children) {
                zipFile(childFile, fileName + "/" + childFile.getName(), zipOut);
            }
            //如果當前遞迴物件是資料夾,加入ZipEntry之後就返回
            return;
        }
        //如果當前的fileToZip不是一個資料夾,是一個檔案,將其以位元組碼形式壓縮到壓縮包裡面
        FileInputStream fis = new FileInputStream(fileToZip);
        ZipEntry zipEntry = new ZipEntry(fileName);
        zipOut.putNextEntry(zipEntry);
        byte[] bytes = new byte[1024];
        int length;
        while ((length = fis.read(bytes)) >= 0) {
            zipOut.write(bytes, 0, length);
        }
        fis.close();
    }
}
  • 要壓縮子目錄及其子目錄檔案,所以需要遞迴遍歷
  • 每次遍歷找到的是目錄時,我們都將其名稱附加“/”,並將其以ZipEntry儲存到壓縮包中,從而保持壓縮的目錄結構。
  • 每次遍歷找到的是檔案時,將其以位元組碼形式壓縮到壓縮包裡面

五、解壓縮zip壓縮包

下面為大家舉例講解解壓縮zip壓縮包。在此示例中,我們將compressed.zip解壓縮到名為unzipTest的新資料夾中。

public class UnzipFile {
    public static void main(String[] args) throws IOException {
        //被解壓的壓縮檔案
        String fileZip = "src/main/resources/unzipTest/compressed.zip";
        //解壓的目標目錄
        File destDir = new File("src/main/resources/unzipTest");

        byte[] buffer = new byte[1024];
        ZipInputStream zis = new ZipInputStream(new FileInputStream(fileZip));
        //獲取壓縮包中的entry,並將其解壓
        ZipEntry zipEntry = zis.getNextEntry();
        while (zipEntry != null) {
            File newFile = newFile(destDir, zipEntry);
            FileOutputStream fos = new FileOutputStream(newFile);
            int len;
            while ((len = zis.read(buffer)) > 0) {
                fos.write(buffer, 0, len);
            }
            fos.close();
            //解壓完成一個entry,再解壓下一個
            zipEntry = zis.getNextEntry();
        }
        zis.closeEntry();
        zis.close();
    }
    //在解壓目標資料夾,新建一個檔案
    public static File newFile(File destinationDir, ZipEntry zipEntry) throws IOException {
        File destFile = new File(destinationDir, zipEntry.getName());

        String destDirPath = destinationDir.getCanonicalPath();
        String destFilePath = destFile.getCanonicalPath();

        if (!destFilePath.startsWith(destDirPath + File.separator)) {
            throw new IOException("該解壓項在目標資料夾之外: " + zipEntry.getName());
        }

        return destFile;
    }
}

歡迎關注我的部落格,裡面有很多精品合集

  • 本文轉載註明出處(必須帶連線,不能只轉文字):字母哥部落格

覺得對您有幫助的話,幫我點贊、分享!您的支援是我不竭的創作動力! 。另外,筆者最近一段時間輸出瞭如下的精品內容,期待您的關注。

相關文章