Java解压和压缩带密码的zip文件

提示:本文介绍的是winzipaes项目,但该开源项目使用起来并不太方便,我最终也没有采用它,如果您有在Java语言环境中处理zip压缩文件的需要,推荐采用zip4j这一开源项目,相比winzipaes仅支持AES算法而言,zip4j支持多种算法,其它方面也是非常优秀,可以说是强大。

详见另一篇文章:

前言

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文件可以使用它解压,没有加密的就需要采取其它方式了。

另外:直接从Google Code上取下来的项目是不支持含有中文的的文件名的,因为winzipaes默认采用的是“ISO-8859-1”编码,但是网上有人稍微修改了一下源码,使其支持中文文件名,请参照这里:

此文就是采用修改后的winzipaes编写,并记录详细修改步骤。

winzipaes项目依赖bcprov的jar包,可以到 上下载。

示例

在研究该项目时写了一个工具类,本来准备用在项目中,最后找到了更好的解决方案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();
   }
  }
 }

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:https://www.heiqu.com/2959b7a20460fcce098f268070ed3188.html