栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

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

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

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

前言

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; 

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包,也可以去那里下载。

修改记录

需要修改的文件有:

  • 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方法中)其它地方不需改动。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持考高分网。

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/137774.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号