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

el/2024/7/24 2:47:12
提示:本文介绍的是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;

public class DecryptionZipUtil {
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);
doZip(sFile, zipFileEncrypter, "", passwd);
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
zipFileEncrypter.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
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);
}
}
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";
List 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();
}
}
}

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 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://blog.csdn.net/zhangyihui1986/article/details/7723649

相关文件下载
 
http://download.csdn.net/detail/zhangyihui1986/4415937

http://www.ngui.cc/el/5557489.html

相关文章

CENTOSnbsp;设置文件夹打开方式:…

1. 系统&#xff0d;&#xff0d;首选项&#xff0d;&#xff0d;文件管理&#xff0d;&#xff0d;行为&#xff0d;&#xff0d;勾选总是在浏览窗口打开。 2.用鼠标滚轮双击文件夹

lucenenbsp;排序、过滤、分词器

1. 排序 1.1. Sort类 public Sort() public Sort(String field) public Sort(String field,Boolean reverse) //默认为false&#xff0c;降序排序 public Sort(String[] fields) public Sort(SortField field) public Sort(SortField[] fields) Sort sortnew Sort(“bookname…

Java判断文件类型nbsp;(转)

通常&#xff0c;在WEB系统中&#xff0c;上传文件时都需要做文件的类型校验&#xff0c;大致有如下几种方法&#xff1a; 1. 通过后缀名&#xff0c;如exe,jpg,bmp,rar,zip等等。 2. 通过读取文件&#xff0c;获取文件的Content-type来判断。 3. 通过读取文件流&#xff0c;根…

java反射调用静态方法

Class c; c Class.forName("class name"); Method m c.getMethod("method name", new Class[] {int.class, int.class, int.class,int.class}); m.invoke(c, new Object[] {1,2, 3, 4});

Cometnbsp;(web技术)

Comet是一种用于web的推送技术&#xff0c;能使服务器能实时地将更新的信息传送到客户端&#xff0c;而无须客户端发出请求&#xff0c;目前有两种实现方式&#xff0c;长轮询和iframe流。 长轮询 长轮询是在打开一条连接以后保持&#xff0c;等待服务器推送来数据再关闭的方式…

CentOS6.2nbsp;解决安装腾达w311m…

解压好软件包;2011_0407_RT3070_RT3370_RT5370_RT5372_Linux_STA_V2.5.0.2_DPO.bz2 在文件夹下make; 提示: make****/lib/modules/2.6.32-279.0.1.el6.x86_64/build没有文件或文件夹。 于是到这个文件夹下&#xff0c;试着打开build&#xff0c;系统提示软链接错误是否删除&…

jsoupnbsp;javanbsp;htmlnbsp;解析器用法

使用选择器语法来查找元素 问题 你想使用类似于CSS或jQuery的语法来查找和操作元素。 方法 可以使用Element.select(String selector) 和 Elements.select(String selector) 方法实现&#xff1a; File input new File("/tmp/input.html"); Document doc Jsoup.par…

centos6.2nbsp;腾达W311M无线USB…

W311M无线USB网卡驱动安装(CentOS) 1.#lsusb //列出挂载的 usb 设备 2.#tar -jxvf 2011_0407_RT3070_RT3370_RT5370_RT5372_Linux_STA_V2.5.0.2_DPO.bz2 //解压驱动文件操作 3.#cd 2011_0407_RT3070_RT3370_RT5370_RT5372_Linux_STA_V2.5.0.2_DPO/ //进入解压文件目录 4.#vi os…

javanbsp;URL相对路径转换成绝对路径

//绝对路径 String absolutePath "http://www.sdfsdfsfdf.com/1/2/3.html"; //相对路径 String relativePath "../../a.jpg"; //以下方法对相对路径进行转换 URL absoluteUrl new URL(absolutePath); URL parseUrl new UR…

Servletnbsp;3.0nbsp;新特性概述

Servlet 3.0 作为 Java EE 6 规范体系中一员&#xff0c;随着 Java EE 6 规范一起发布。该版本在前一版本&#xff08;Servlet 2.5&#xff09;的基础上提供了若干新特性用于简化 Web 应用的开发和部署。其中有几项特性的引入让开发者感到非常兴奋&#xff0c;同时也获得了 Jav…