时间:2021-05-20
前言
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文件可以使用它解压,没有加密的就需要采取其它方式了。
此文就是采用修改后的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(); } } } /** * 测试 * @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包,也可以去那里下载。
修改记录
需要修改的文件有:
在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方法基础上增加一个重载方法:
其实就是增加一个参数并增加了下面这段代码:
if (utf8Flag) {this.flag |=(1 << 11);}当然不要忘了将调用该方法地方修改一下,传进utf8Flag参数
AesZipFileEncrypter类里有两处(在两个add方法中)其它地方不需改动。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
ZIP4j压缩与解压的实例详解使用的jar包:zip4j_1.3.2.jar基本功能:针对ZIP压缩文件创建、添加、分卷、更新和移除文件(读写有密码保护的Zip
Java解压缩zip-多个文件(包括文件夹),具体如下:对多个文件和文件夹进行压缩,对复杂的文件目录进行解压。压缩方法使用的是可变参数,可以压缩1到多个文件..
Java解压缩zip-多个文件(包括文件夹),对多个文件和文件夹进行压缩,对复杂的文件目录进行解压。压缩方法使用的是可变参数,可以压缩1到多个文件..可以写数组
zip文件是压缩文件,解压即可安装。zip压缩包的解压步骤是: 1、在电脑正常安装解压缩软件的前提下,右键选择解压。 2、将该文件解压到指定位置后,点击确定
前言最近的项目中需要用到解压缩包的功能,客户给出的压缩包的格式主要是rar和zip,因此就打算使用java调用7zip的命令行进行解压文件,本文主要记录一下实现