歡迎來到Linux教程網
Linux教程網
Linux教程網
Linux教程網
您现在的位置: Linux教程網 >> UnixLinux >  >> Linux編程 >> Linux編程

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

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

詳見另一篇文章: http://www.linuxidc.com/Linux/2014-11/109061.htm

前言

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://www.linuxidc.com/Linux/2014-11/109060.htm

此文就是采用修改後的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://www.linuxidc.com/Linux/2014-11/109060.htm

需要修改的文件有:

  • 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 integritiy

 this.flag |= 1; // bit0 - encrypted
 if (utf8Flag) {
  this.flag |=(1 << 11);
 }
 // flag |= 8; // bit3 - use data descriptor
 this.primaryCompressionMethod = 0x63;

 byte[] extraBytes = new byte[11];
 extraBytes = new byte[11];

 // extra data header ID for AES encryption is 0x9901
 extraBytes[0] = 0x01;
 extraBytes[1] = (byte)0x99;

 // data size (currently 7, but subject to possible increase in the
 // future)
 extraBytes[2] = 0x07; // data size
 extraBytes[3] = 0x00; // data size
 // Integer version number specific to the zip vendor
 extraBytes[4] = 0x02; // version number
 extraBytes[5] = 0x00; // version number

 // 2-character vendor ID
 extraBytes[6] = 0x41; // vendor id
 extraBytes[7] = 0x45; // vendor id

 // AES encryption strength - 1=128, 2=192, 3=256
 extraBytes[8] = 0x03;

 // actual compression method - 0x0000==stored (no compression) - 2 bytes
 extraBytes[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://www.linuxidc.com/Linux/2014-11/109059.htm

相關文件下載

------------------------------------------分割線------------------------------------------

免費下載地址在 http://linux.linuxidc.com/

用戶名與密碼都是www.linuxidc.com

具體下載目錄在 /2014年資料/11月/6日/Java解壓和壓縮帶密碼的zip文件

下載方法見 http://www.linuxidc.com/Linux/2013-07/87684.htm

------------------------------------------分割線------------------------------------------

Copyright © Linux教程網 All Rights Reserved