Java解壓和壓縮帶密碼的zip文件過程詳解


 

https://www.jb51.net/article/164137.htm

前言

JDK自帶的ZIP操作接口(java.util.zip包,請參看文章末尾的博客鏈接)並不支持密碼,甚至也不支持中文文件名。

為了解決ZIP壓縮文件的密碼問題,在網上搜索良久,終於找到了winzipaes開源項目。

該項目在google code下托管 ,僅支持AES壓縮和解壓zip文件( This library only supports Win-Zip's 256-Bit AES mode.)。網站上下載的文件是源代碼,最新版本為winzipaes_src_20120416.zip,本示例就是在此基礎上編寫。

詳述

項目使用很簡單,利用源碼自己導出一個jar文件,在項目中引用即可。

這里有一個需要注意的問題,就是如果給定ZIP文件沒有密碼,那么就不能使用該項目解壓,如果壓縮文件沒有密碼卻使用該項目解壓在這里會報一個異常,所以使用中需要注意:加密ZIP文件可以使用它解壓,沒有加密的就需要采取其它方式了。

此文就是采用修改后的winzipaes編寫,並記錄詳細修改步驟。

winzipaes項目依賴bcprov的jar包

 

Windows命令行:

 1   try {
 2 
 3    String cmd = "unzip -o -P" + passWord + nssDecomFilePath + "\\"
 4      + zipFileName;
 5 
 6    Runtime.getRuntime().exec(cmd);
 7   } catch (Exception ex) {
 8    return false;
 9   }

 

示例

在研究該項目時寫了一個工具類,本來准備用在項目中,最后找到了更好的解決方案zip4j來代替,所以最終沒有采用。

  1 package com.ninemax.demo.zip.decrypt;
  2 import java.io.File;
  3 import java.io.IOException;
  4 import java.util.List;
  5 import java.util.zip.DataFormatException;
  6 import org.apache.commons.io.FileUtils;
  7 import de.idyl.winzipaes.AesZipFileDecrypter;
  8 import de.idyl.winzipaes.AesZipFileEncrypter;
  9 import de.idyl.winzipaes.impl.AESDecrypter;
 10 import de.idyl.winzipaes.impl.AESDecrypterBC;
 11 import de.idyl.winzipaes.impl.AESEncrypter;
 12 import de.idyl.winzipaes.impl.AESEncrypterBC;
 13 import de.idyl.winzipaes.impl.ExtZipEntry; 
 14 /**
 15  * 壓縮指定文件或目錄為ZIP格式壓縮文件
 16  * 支持中文(修改源碼后)
 17  * 支持密碼(僅支持256bit的AES加密解密)
 18  * 依賴bcprov項目(bcprov-jdk16-140.jar)
 19  * 
 20  * @author zyh
 21  */
 22 public class DecryptionZipUtil {    
 23     /**
 24      * 使用指定密碼將給定文件或文件夾壓縮成指定的輸出ZIP文件
 25      * @param srcFile 需要壓縮的文件或文件夾
 26      * @param destPath 輸出路徑
 27      * @param passwd 壓縮文件使用的密碼
 28      */
 29     public static void zip(String srcFile,String destPath,String passwd) {
 30         AESEncrypter encrypter = new AESEncrypterBC();
 31         AesZipFileEncrypter zipFileEncrypter = null;
 32         try {
 33             zipFileEncrypter = new AesZipFileEncrypter(destPath, encrypter);
 34             /**
 35              * 此方法是修改源碼后添加,用以支持中文文件名
 36              */
 37             zipFileEncrypter.setEncoding("utf8");
 38             File sFile = new File(srcFile);
 39             /**
 40              * AesZipFileEncrypter提供了重載的添加Entry的方法,其中:
 41              * add(File f, String passwd) 
 42              *             方法是將文件直接添加進壓縮文件
 43              * 
 44              * add(File f, String pathForEntry, String passwd)
 45              *             方法是按指定路徑將文件添加進壓縮文件
 46              * pathForEntry - to be used for addition of the file (path within zip file)
 47              */
 48             doZip(sFile, zipFileEncrypter, "", passwd);
 49         } catch (IOException e) {
 50             e.printStackTrace();
 51         } finally {
 52             try {
 53                 zipFileEncrypter.close();
 54             } catch (IOException e) {
 55                 e.printStackTrace();
 56             }
 57         }
 58     }
 59     
 60     /**
 61      * 具體壓縮方法,將給定文件添加進壓縮文件中,並處理壓縮文件中的路徑
 62      * @param file 給定磁盤文件(是文件直接添加,是目錄遞歸調用添加)
 63      * @param encrypter AesZipFileEncrypter實例,用於輸出加密ZIP文件
 64      * @param pathForEntry ZIP文件中的路徑
 65      * @param passwd 壓縮密碼
 66      * @throws IOException
 67      */
 68     private static void doZip(File file, AesZipFileEncrypter encrypter,
 69             String pathForEntry, String passwd) throws IOException {
 70         if (file.isFile()) {
 71             pathForEntry += file.getName();
 72             encrypter.add(file, pathForEntry, passwd);
 73             return;
 74         }
 75         pathForEntry += file.getName() + File.separator;
 76         for(File subFile : file.listFiles()) {
 77             doZip(subFile, encrypter, pathForEntry, passwd);
 78         }
 79     }
 80     
 81     /**
 82      * 使用給定密碼解壓指定壓縮文件到指定目錄
 83      * @param inFile 指定Zip文件
 84      * @param outDir 解壓目錄
 85      * @param passwd 解壓密碼
 86      */
 87     public static void unzip(String inFile, String outDir, String passwd) {
 88         File outDirectory = new File(outDir);
 89         if (!outDirectory.exists()) {
 90             outDirectory.mkdir();
 91         }
 92         AESDecrypter decrypter = new AESDecrypterBC();
 93         AesZipFileDecrypter zipDecrypter = null;
 94         try {
 95             zipDecrypter = new AesZipFileDecrypter(new File(inFile), decrypter);
 96             AesZipFileDecrypter.charset = "utf-8";
 97             /**
 98              * 得到ZIP文件中所有Entry,但此處好像與JDK里不同,目錄不視為Entry
 99              * 需要創建文件夾,entry.isDirectory()方法同樣不適用,不知道是不是自己使用錯誤
100              * 處理文件夾問題處理可能不太好
101              */
102             List<ExtZipEntry> entryList = zipDecrypter.getEntryList();
103             for(ExtZipEntry entry : entryList) {
104                 String eName = entry.getName();
105                 String dir = eName.substring(0, eName.lastIndexOf(File.separator) + 1);
106                 File extractDir = new File(outDir, dir);
107                 if (!extractDir.exists()) {
108                     FileUtils.forceMkdir(extractDir);
109                 }
110                 /**
111                  * 抽出文件
112                  */
113                 File extractFile = new File(outDir + File.separator + eName);
114                 zipDecrypter.extractEntry(entry, extractFile, passwd);
115             }
116         } catch (IOException e) {
117             e.printStackTrace();
118         } catch (DataFormatException e) {
119             e.printStackTrace();
120         } finally {
121             try {
122                 zipDecrypter.close();
123             } catch (IOException e) {
124                 e.printStackTrace();
125             }
126         }
127     } 
128     /**
129      * 測試
130      * @param args
131      */
132     public static void main(String[] args) {
133         /**
134          * 壓縮測試
135          * 可以傳文件或者目錄
136          */
137 //        zip("M:\\ZIP\\test\\bb\\a\\t.txt", "M:\\ZIP\\test\\temp1.zip", "zyh");
138 //        zip("M:\\ZIP\\test\\bb", "M:\\ZIP\\test\\temp2.zip", "zyh");        
139         unzip("M:\\ZIP\\test\\temp2.zip", "M:\\ZIP\\test\\temp", "zyh");
140     }
141 }

壓縮多個文件時,有兩個方法(第一種沒試):

(1) 預先把多個文件壓縮成zip,然后調用enc.addAll(inZipFile, password);方法將多個zip文件加進來。

(2)針對需要壓縮的文件循環調用enc.add(inFile, password);,每次都用相同的密碼。

修改源碼后的項目可到上面提到的博客去下載,或者參照博客自己修改,其實也很容易,畢竟只有幾處改動。

另外我的CSDN下載頻道也上傳了修改后的源碼和jar包,也可以去那里下載。

修改記錄

需要修改的文件有:

  • ExtZipOutputStream
  • ExtZipEntry
  • AesZipFileEncrypter

在ExtZipOutputStream里增加一成員變量並添加兩個方法:

 1 protected String encoding = "iso-8859-1";    
 2 public boolean utf8Flg = false;
 3     public void setEncoding(String encoding) {
 4         this.encoding = encoding;
 5         utf8Flg |= isUTF8(encoding);
 6     }
 7     protected boolean isUTF8(String encoding) {
 8     if (encoding == null) {
 9       // check platform's default encoding
10       encoding = System.getProperty("file.encoding");
11     }
12     return "UTF8".equalsIgnoreCase(encoding)
13       || "UTF-8".equalsIgnoreCase(encoding);
14   }

然后將ExtZipOutputStream的(134行和158行左右)iso-8859-1編碼替換成上面設置的編碼格式 

接着,再將106行左右文件名長度取得代碼改成:

writeShort(entry.getName().getBytes(encoding).length); // file name length

這里有個地方需要注意,當文件名是utf8編碼格式的時候,需要設置Zip包的通用位標志 (不明白)

第十一個比特為1,代碼修改如下: 

修改ExtZipEntry類在initEncryptedEntry方法基礎上增加一個重載方法:

 1 public void initEncryptedEntry(boolean utf8Flag) {
 2     setCrc(0); // CRC-32 / for encrypted files it's 0 as AES/MAC checks integritiy 
 3     this.flag |= 1; // bit0 - encrypted
 4     if (utf8Flag) {
 5         this.flag |=(1 << 11);
 6     }
 7     // flag |= 8; // bit3 - use data descriptor
 8     this.primaryCompressionMethod = 0x63;
 9  
10     byte[] extraBytes = new byte[11];
11     extraBytes = new byte[11];
12     // extra data header ID for AES encryption is 0x9901
13     extraBytes[0] = 0x01;
14     extraBytes[1] = (byte)0x99; 
15     // data size (currently 7, but subject to possible increase in the
16     // future)
17     extraBytes[2] = 0x07; // data size
18     extraBytes[3] = 0x00; // data size
19     // Integer version number specific to the zip vendor
20     extraBytes[4] = 0x02; // version number
21     extraBytes[5] = 0x00; // version number
22  
23     // 2-character vendor ID
24     extraBytes[6] = 0x41; // vendor id
25     extraBytes[7] = 0x45; // vendor id 
26     // AES encryption strength - 1=128, 2=192, 3=256
27     extraBytes[8] = 0x03; 
28     // actual compression method - 0x0000==stored (no compression) - 2 bytes
29     extraBytes[9] = (byte) (getMethod() & 0xff);
30     extraBytes[10] = (byte) ((getMethod() & 0xff00) >> 8);
31  
32     setExtra(extraBytes);
33 }

其實就是增加一個參數並增加了下面這段代碼:

1 if (utf8Flag) {
2   this.flag |=(1 << 11);
3 }

 

 

當然不要忘了將調用該方法地方修改一下,傳進utf8Flag參數

AesZipFileEncrypter類里有兩處(在兩個add方法中)其它地方不需改動。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM