Java解压和压缩带密码的zip文件过程详解
前言
JDK自带的ZIP操作接口(java.util.zip包,请参看文章末尾的博客链接)并不支持密码,甚至也不支持中文文件名。
为了解决ZIP压缩文件的密码问题,在网上搜索良久,终于找到了winzipaes开源项目。
该项目在googlecode下托管 ,仅支持AES压缩和解压zip文件( ThislibraryonlysupportsWin-Zip's256-BitAESmode.)。网站上下载的文件是源代码,最新版本为winzipaes_src_20120416.zip,本示例就是在此基础上编写。
详述
项目使用很简单,利用源码自己导出一个jar文件,在项目中引用即可。
这里有一个需要注意的问题,就是如果给定ZIP文件没有密码,那么就不能使用该项目解压,如果压缩文件没有密码却使用该项目解压在这里会报一个异常,所以使用中需要注意:加密ZIP文件可以使用它解压,没有加密的就需要采取其它方式了。
此文就是采用修改后的winzipaes编写,并记录详细修改步骤。
winzipaes项目依赖bcprov的jar包
示例
在研究该项目时写了一个工具类,本来准备用在项目中,最后找到了更好的解决方案zip4j来代替,所以最终没有采用。
packagecom.ninemax.demo.zip.decrypt;
importjava.io.File;
importjava.io.IOException;
importjava.util.List;
importjava.util.zip.DataFormatException;
importorg.apache.commons.io.FileUtils;
importde.idyl.winzipaes.AesZipFileDecrypter;
importde.idyl.winzipaes.AesZipFileEncrypter;
importde.idyl.winzipaes.impl.AESDecrypter;
importde.idyl.winzipaes.impl.AESDecrypterBC;
importde.idyl.winzipaes.impl.AESEncrypter;
importde.idyl.winzipaes.impl.AESEncrypterBC;
importde.idyl.winzipaes.impl.ExtZipEntry;
/**
*压缩指定文件或目录为ZIP格式压缩文件
*支持中文(修改源码后)
*支持密码(仅支持256bit的AES加密解密)
*依赖bcprov项目(bcprov-jdk16-140.jar)
*
*@authorzyh
*/
publicclassDecryptionZipUtil{
/**
*使用指定密码将给定文件或文件夹压缩成指定的输出ZIP文件
*@paramsrcFile需要压缩的文件或文件夹
*@paramdestPath输出路径
*@parampasswd压缩文件使用的密码
*/
publicstaticvoidzip(StringsrcFile,StringdestPath,Stringpasswd){
AESEncrypterencrypter=newAESEncrypterBC();
AesZipFileEncrypterzipFileEncrypter=null;
try{
zipFileEncrypter=newAesZipFileEncrypter(destPath,encrypter);
/**
*此方法是修改源码后添加,用以支持中文文件名
*/
zipFileEncrypter.setEncoding("utf8");
FilesFile=newFile(srcFile);
/**
*AesZipFileEncrypter提供了重载的添加Entry的方法,其中:
*add(Filef,Stringpasswd)
* 方法是将文件直接添加进压缩文件
*
*add(Filef,StringpathForEntry,Stringpasswd)
* 方法是按指定路径将文件添加进压缩文件
*pathForEntry-tobeusedforadditionofthefile(pathwithinzipfile)
*/
doZip(sFile,zipFileEncrypter,"",passwd);
}catch(IOExceptione){
e.printStackTrace();
}finally{
try{
zipFileEncrypter.close();
}catch(IOExceptione){
e.printStackTrace();
}
}
}
/**
*具体压缩方法,将给定文件添加进压缩文件中,并处理压缩文件中的路径
*@paramfile给定磁盘文件(是文件直接添加,是目录递归调用添加)
*@paramencrypterAesZipFileEncrypter实例,用于输出加密ZIP文件
*@parampathForEntryZIP文件中的路径
*@parampasswd压缩密码
*@throwsIOException
*/
privatestaticvoiddoZip(Filefile,AesZipFileEncrypterencrypter,
StringpathForEntry,Stringpasswd)throwsIOException{
if(file.isFile()){
pathForEntry+=file.getName();
encrypter.add(file,pathForEntry,passwd);
return;
}
pathForEntry+=file.getName()+File.separator;
for(FilesubFile:file.listFiles()){
doZip(subFile,encrypter,pathForEntry,passwd);
}
}
/**
*使用给定密码解压指定压缩文件到指定目录
*@paraminFile指定Zip文件
*@paramoutDir解压目录
*@parampasswd解压密码
*/
publicstaticvoidunzip(StringinFile,StringoutDir,Stringpasswd){
FileoutDirectory=newFile(outDir);
if(!outDirectory.exists()){
outDirectory.mkdir();
}
AESDecrypterdecrypter=newAESDecrypterBC();
AesZipFileDecrypterzipDecrypter=null;
try{
zipDecrypter=newAesZipFileDecrypter(newFile(inFile),decrypter);
AesZipFileDecrypter.charset="utf-8";
/**
*得到ZIP文件中所有Entry,但此处好像与JDK里不同,目录不视为Entry
*需要创建文件夹,entry.isDirectory()方法同样不适用,不知道是不是自己使用错误
*处理文件夹问题处理可能不太好
*/
ListentryList=zipDecrypter.getEntryList();
for(ExtZipEntryentry:entryList){
StringeName=entry.getName();
Stringdir=eName.substring(0,eName.lastIndexOf(File.separator)+1);
FileextractDir=newFile(outDir,dir);
if(!extractDir.exists()){
FileUtils.forceMkdir(extractDir);
}
/**
*抽出文件
*/
FileextractFile=newFile(outDir+File.separator+eName);
zipDecrypter.extractEntry(entry,extractFile,passwd);
}
}catch(IOExceptione){
e.printStackTrace();
}catch(DataFormatExceptione){
e.printStackTrace();
}finally{
try{
zipDecrypter.close();
}catch(IOExceptione){
e.printStackTrace();
}
}
}
/**
*测试
*@paramargs
*/
publicstaticvoidmain(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包,也可以去那里下载。
修改记录
需要修改的文件有:
- ExtZipOutputStream
- ExtZipEntry
- AesZipFileEncrypter
在ExtZipOutputStream里增加一成员变量并添加两个方法:
protectedStringencoding="iso-8859-1";
publicbooleanutf8Flg=false;
publicvoidsetEncoding(Stringencoding){
this.encoding=encoding;
utf8Flg|=isUTF8(encoding);
}
protectedbooleanisUTF8(Stringencoding){
if(encoding==null){
//checkplatform'sdefaultencoding
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);//filenamelength
这里有个地方需要注意,当文件名是utf8编码格式的时候,需要设置Zip包的通用位标志(不明白)
第十一个比特为1,代码修改如下:
修改ExtZipEntry类在initEncryptedEntry方法基础上增加一个重载方法:
publicvoidinitEncryptedEntry(booleanutf8Flag){
setCrc(0);//CRC-32/forencryptedfilesit's0asAES/MACchecksintegritiy
this.flag|=1;//bit0-encrypted
if(utf8Flag){
this.flag|=(1<<11);
}
//flag|=8;//bit3-usedatadescriptor
this.primaryCompressionMethod=0x63;
byte[]extraBytes=newbyte[11];
extraBytes=newbyte[11];
//extradataheaderIDforAESencryptionis0x9901
extraBytes[0]=0x01;
extraBytes[1]=(byte)0x99;
//datasize(currently7,butsubjecttopossibleincreaseinthe
//future)
extraBytes[2]=0x07;//datasize
extraBytes[3]=0x00;//datasize
//Integerversionnumberspecifictothezipvendor
extraBytes[4]=0x02;//versionnumber
extraBytes[5]=0x00;//versionnumber
//2-charactervendorID
extraBytes[6]=0x41;//vendorid
extraBytes[7]=0x45;//vendorid
//AESencryptionstrength-1=128,2=192,3=256
extraBytes[8]=0x03;
//actualcompressionmethod-0x0000==stored(nocompression)-2bytes
extraBytes[9]=(byte)(getMethod()&0xff);
extraBytes[10]=(byte)((getMethod()&0xff00)>>8);
setExtra(extraBytes);
}
其实就是增加一个参数并增加了下面这段代码:
if(utf8Flag){
this.flag|=(1<<11);
}
当然不要忘了将调用该方法地方修改一下,传进utf8Flag参数
AesZipFileEncrypter类里有两处(在两个add方法中)其它地方不需改动。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。