Java解壓和壓縮帶密碼的zip檔案

來源:互聯網
上載者:User

提示:本文介紹的是winzipaes項目,但該開源項目使用起來並不太方便,我最終也沒有採用它,如果您有在Java語言環境中處理zip壓縮檔的需要,推薦採用zip4j這一開源項目,相比winzipaes僅支援AES演算法而言,zip4j支援多種演算法,其它方面也是非常優秀,可以說是強大。

詳見另一篇部落格:

  • http://blog.csdn.net/zhangyihui1986/article/details/7921376

前言

JDK內建的ZIP操作介面(java.util.zip包,請參看文章末尾的部落格連結)並不支援密碼,甚至也不支援中文檔案名稱。

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

該項目在google code下託管,網址為:http://code.google.com/p/winzipaes ,僅支援AES壓縮和解壓zip檔案( This
library only supports Win-Zip's 256-Bit AES mode.)。網站上下載的檔案是原始碼,最新版本為winzipaes_src_20120416.zip,本樣本就是在此基礎上編寫。

詳述

項目使用很簡單,利用源碼自己匯出一個jar檔案,在項目中引用即可。

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

另外:直接從Google Code上取下來的項目是不支援含有中文的的檔案名稱的,因為winzipaes預設採用的是“ISO-8859-1”編碼,但是網上有人稍微修改了一下源碼,使其支援中文檔案名稱,請參照這裡:http://xjlsgcjdtc.iteye.com/blog/1439514

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

winzipaes項目依賴bcprov的jar包,可以到 http://www.bouncycastle.org/java.html 上下載。

樣本

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

package com.ninemax.demo.zip.decrypt;import java.io.File;import java.io.IOException;import java.util.List;import java.util.zip.DataFormatException;import org.apache.commons.io.FileUtils;import de.idyl.winzipaes.AesZipFileDecrypter;import de.idyl.winzipaes.AesZipFileEncrypter;import de.idyl.winzipaes.impl.AESDecrypter;import de.idyl.winzipaes.impl.AESDecrypterBC;import de.idyl.winzipaes.impl.AESEncrypter;import de.idyl.winzipaes.impl.AESEncrypterBC;import de.idyl.winzipaes.impl.ExtZipEntry;/** * 壓縮指定檔案或目錄為ZIP格式壓縮檔 * 支援中文(修改源碼後) * 支援密碼(僅支援256bit的AES加密解密) * 依賴bcprov項目(bcprov-jdk16-140.jar) *  * @author zyh */public class DecryptionZipUtil {/** * 使用指定密碼將給定檔案或檔案夾壓縮成指定的輸出ZIP檔案 * @param srcFile 需要壓縮的檔案或檔案夾 * @param destPath 輸出路徑 * @param passwd 壓縮檔使用的密碼 */public static void zip(String srcFile,String destPath,String passwd) {AESEncrypter encrypter = new AESEncrypterBC();AesZipFileEncrypter zipFileEncrypter = null;try {zipFileEncrypter = new AesZipFileEncrypter(destPath, encrypter);/** * 此方法是修改源碼後添加,用以支援中文檔案名稱 */zipFileEncrypter.setEncoding("utf8");File sFile = new File(srcFile);/** * AesZipFileEncrypter提供了重載的添加Entry的方法,其中: * add(File f, String passwd)  * 方法是將檔案直接添加進壓縮檔 *  * add(File f,  String pathForEntry, String passwd) * 方法是按指定路徑將檔案添加進壓縮檔 * pathForEntry - to be used for addition of the file (path within zip file) */doZip(sFile, zipFileEncrypter, "", passwd);} catch (IOException e) {e.printStackTrace();} finally {try {zipFileEncrypter.close();} catch (IOException e) {e.printStackTrace();}}}/** * 具體壓縮方法,將給定檔案添加進壓縮檔中,並處理壓縮檔中的路徑 * @param file 給定磁碟檔案(是檔案直接添加,是目錄遞迴調用添加) * @param encrypter AesZipFileEncrypter執行個體,用於輸出加密ZIP檔案 * @param pathForEntry ZIP檔案中的路徑 * @param passwd 壓縮密碼 * @throws IOException */private static void doZip(File file, AesZipFileEncrypter encrypter,String pathForEntry, String passwd) throws IOException {if (file.isFile()) {pathForEntry += file.getName();encrypter.add(file, pathForEntry, passwd);return;}pathForEntry += file.getName() + File.separator;for(File subFile : file.listFiles()) {doZip(subFile, encrypter, pathForEntry, passwd);}}/** * 使用給定密碼解壓指定壓縮檔到指定目錄 * @param inFile 指定Zip檔案 * @param outDir 解壓目錄 * @param passwd 解壓密碼 */public static void unzip(String inFile, String outDir, String passwd) {File outDirectory = new File(outDir);if (!outDirectory.exists()) {outDirectory.mkdir();}AESDecrypter decrypter = new AESDecrypterBC();AesZipFileDecrypter zipDecrypter = null;try {zipDecrypter = new AesZipFileDecrypter(new File(inFile), decrypter);AesZipFileDecrypter.charset = "utf-8";/** * 得到ZIP檔案中所有Entry,但此處好像與JDK裡不同,目錄不視為Entry * 需要建立檔案夾,entry.isDirectory()方法同樣不適用,不知道是不是自己使用錯誤 * 處理檔案夾問題處理可能不太好 */List<ExtZipEntry> entryList = zipDecrypter.getEntryList();for(ExtZipEntry entry : entryList) {String eName = entry.getName();String dir = eName.substring(0, eName.lastIndexOf(File.separator) + 1);File extractDir = new File(outDir, dir);if (!extractDir.exists()) {FileUtils.forceMkdir(extractDir);}/** * 抽出檔案 */File extractFile = new File(outDir + File.separator + eName);zipDecrypter.extractEntry(entry, extractFile, passwd);}} catch (IOException e) {e.printStackTrace();} catch (DataFormatException e) {e.printStackTrace();} finally {try {zipDecrypter.close();} catch (IOException e) {e.printStackTrace();}}}/** * 測試 * @param args */public static void main(String[] args) {/** * 壓縮測試 * 可以傳檔案或者目錄 *///zip("M:\\ZIP\\test\\bb\\a\\t.txt", "M:\\ZIP\\test\\temp1.zip", "zyh");//zip("M:\\ZIP\\test\\bb", "M:\\ZIP\\test\\temp2.zip", "zyh");unzip("M:\\ZIP\\test\\temp2.zip", "M:\\ZIP\\test\\temp", "zyh");}}

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

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

(2)針對需要壓縮的檔案迴圈調用enc.add(inFile, password);,每次都用相同的密碼。


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

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

修改記錄

參考http://xjlsgcjdtc.iteye.com/blog/1439514

需要修改的檔案有:
  • ExtZipOutputStream
  • ExtZipEntry
  • AesZipFileEncrypter

在ExtZipOutputStream裡增加一成員變數並添加兩個方法:

protected String encoding = "iso-8859-1";public boolean utf8Flg = false;

public void setEncoding(String encoding) {this.encoding = encoding;utf8Flg |= isUTF8(encoding);}
protected boolean isUTF8(String encoding) {        if (encoding == null) {            // check platform's default encoding            encoding = System.getProperty("file.encoding");        }        return "UTF8".equalsIgnoreCase(encoding)            || "UTF-8".equalsIgnoreCase(encoding);    }

然後將ExtZipOutputStream的(134行和158行左右)iso-8859-1編碼替換成上面設定的編碼格式 
接著,再將106行左右檔案名稱長度取得代碼改成:

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

這裡有個地方需要注意,當檔案名稱是utf8編碼格式的時候,需要設定Zip包的通用位標誌 (不明白)
第十一個位元為1,代碼修改如下: 
修改ExtZipEntry類在initEncryptedEntry方法基礎上增加一個重載方法:

public void initEncryptedEntry(boolean utf8Flag) {setCrc(0); // CRC-32 / for encrypted files it's 0 as AES/MAC checks integritiythis.flag |= 1; // bit0 - encryptedif (utf8Flag) {this.flag |=(1 << 11);}// flag |= 8; // bit3 - use data descriptorthis.primaryCompressionMethod = 0x63;byte[] extraBytes = new byte[11];extraBytes = new byte[11];// extra data header ID for AES encryption is 0x9901extraBytes[0] = 0x01;extraBytes[1] = (byte)0x99;// data size (currently 7, but subject to possible increase in the// future)extraBytes[2] = 0x07; // data sizeextraBytes[3] = 0x00; // data size// Integer version number specific to the zip vendorextraBytes[4] = 0x02; // version numberextraBytes[5] = 0x00; // version number// 2-character vendor IDextraBytes[6] = 0x41; // vendor idextraBytes[7] = 0x45; // vendor id// AES encryption strength - 1=128, 2=192, 3=256extraBytes[8] = 0x03;// actual compression method - 0x0000==stored (no compression) - 2 bytesextraBytes[9] = (byte) (getMethod() & 0xff);extraBytes[10] = (byte) ((getMethod() & 0xff00) >> 8);setExtra(extraBytes);}

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

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

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

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


註:以上代碼我自己已測試通過,如果哪位朋友測試出錯,請留言!


Java基本JDK壓縮和解壓ZIP檔案請參照:

http://blog.csdn.net/zhangyihui1986/article/details/7723649

相關檔案下載

http://download.csdn.net/detail/zhangyihui1986/4415937

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.