java中ant包中的org.apache.tools.zip实现压缩和解压缩

本文介绍如何利用 Apache Ant 中的 org.apache.tools.zip 库来解决 Java 标准库在处理含有中文文件名的 ZIP 文件时遇到的乱码问题。提供了压缩和解压缩的示例代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

其实apache中的ant包(请自行GOOGLE之ant.jar)中有一个更好的类,已经支持中文了,我们就不重复制造轮子了,拿来用吧,
这里最主要的功能是实现了可以指定多个文件到同一个压缩包的功能

用org.apache.tools.zip压缩/解压缩zip文件的例子,用来解决中文乱码问题。

importjava.io.BufferedInputStream;
importjava.io.BufferedOutputStream;
importjava.io.File;
importjava.io.FileInputStream;
importjava.io.FileNotFoundException;
importjava.io.FileOutputStream;
importjava.io.IOException;
importjava.io.InputStream;
importjava.util.ArrayList;
importjava.util.List;
importjava.util.zip.CRC32;
importjava.util.zip.CheckedOutputStream;
importjava.util.zip.Deflater;

importorg.apache.tools.zip.ZipEntry;
importorg.apache.tools.zip.ZipOutputStream;

/**
*功能:使用ApacheAnt里提供的org.apache.tools.zip实现zip压缩和解压(支持中文文件名)
*解决了由于java.util.zip包不支持汉字的问题。使用java.util.zip包时,当zip文件中有名字为中文的文件时,
*就会出现异常:"Exceptioninthread"main"java.lang.IllegalArgumentExceptionat
*java.util.zip.ZipInputStream.getUTF8String(ZipInputStream.java:285)
*
*@author夏明龙E-mail:邮箱
*@version创建时间:2013-3-22上午10:40:21类说明:
*/
publicclassAntZipUtil{
privatestaticListlist=newArrayList();

privatestaticListlistFile(Stringpath){
Filefile=newFile(path);
String[]array=null;
StringsTemp="";

if(!file.isDirectory()){
returnnull;
}
array=file.list();
if(array.length>0){
for(inti=0;i<array.length;i++){
sTemp=path+array[i];
file=newFile(sTemp);
if(file.isDirectory()){
listFile(sTemp+"/");
}else
list.add(sTemp);
}
}else{
returnnull;
}

returnlist;
}

publicstaticvoidzip(Stringneedtozipfilepath,Stringzipfilepath){
try{
byte[]b=newbyte[512];

Fileneedtozipfile=newFile(needtozipfilepath);

if(!needtozipfile.exists()){
System.err.println("指定的要压缩的文件或目录不存在.");
return;
}

StringzipFile=zipfilepath;
FiletargetFile=newFile(zipFile.substring(0,zipFile.indexOf("\\")+1));

if(!targetFile.exists()){
System.out.println("指定的目标文件或目录不存在.");
return;
}

Stringfilepath=needtozipfilepath;
ListfileList=listFile(filepath);
FileOutputStreamfileOutputStream=newFileOutputStream(zipFile);
CheckedOutputStreamcs=newCheckedOutputStream(fileOutputStream,newCRC32());
ZipOutputStreamout=newZipOutputStream(newBufferedOutputStream(cs));

for(inti=0;i<fileList.size();i++){
InputStreamin=newFileInputStream((String)fileList.get(i));
StringfileName=((String)fileList.get(i)).replace(File.separatorChar,'/');
fileName=fileName.substring(fileName.indexOf("/")+1);
ZipEntrye=newZipEntry(fileName);
out.putNextEntry(e);
intlen=0;
while((len=in.read(b))!=-1){
out.write(b,0,len);
}
out.closeEntry();
}
out.close();
}catch(Exceptione){
e.printStackTrace();
}
}

/////////////////////////////////////////
/**
*压缩文件或者文件夹
*
*@parambaseDirName
*压缩的根目录
*@paramfileName
*根目录下待压缩的文件或文件夹名
*@paramtargetFileName
*目标ZIP文件星号"*"表示压缩根目录下的全部文件
*
*/
publicstaticbooleanzip(StringbaseDirName,String[]fileNames,
StringtargetFileName,Stringencoding){
booleanflag=false;
try{
//判断"压缩的根目录"是否存在!是否是一个文件夹!
FilebaseDir=newFile(baseDirName);
if(!baseDir.exists()||(!baseDir.isDirectory())){
System.err.println("压缩失败!根目录不存在:"+baseDirName);
returnfalse;
}

//得到这个"压缩的根目录"的绝对路径
StringbaseDirPath=baseDir.getAbsolutePath();

//由这个"目标ZIP文件"文件名得到一个压缩对象ZipOutputStream
FiletargetFile=newFile(targetFileName);
ZipOutputStreamout=newZipOutputStream(newFileOutputStream(
targetFile));
//中文有乱码,引进下面的改造类
//CnZipOutputStreamout=newCnZipOutputStream(new
//FileOutputStream(targetFile),encoding);

//设置压缩编码ApacheAnt有个包专门处理ZIP文件,可以指定文件名的编码方式。由此可以解决问题。例如:用
//org.apache.tools.zip.ZipOutputStream代替java.util.zip.ZipOutputStream。ZipOutputStream
//out=.....;out.setEncoding("GBK");
//out.setEncoding("GBK");//设置为GBK后在windows下就不会乱码了,如果要放到Linux或者Unix下就不要设置了
out.setEncoding(encoding);

//"*"表示压缩包括根目录baseDirName在内的全部文件到targetFileName文件下
if(fileNames.equals("*")){
AntZipUtil.dirToZip(baseDirPath,baseDir,out);
}else{
File[]files=newFile[fileNames.length];
for(inti=0;i<files.length;i++){
//根据parent抽象路径名和child路径名字符串创建一个新File实例。
files[i]=newFile(baseDir,fileNames[i]);
}
if(files[0].isFile()){
//调用本类的一个静态方法压缩一个文件
//CompressUtil.fileToZip(baseDirPath,file,out);
AntZipUtil.filesToZip(baseDirPath,files,out);
}

}
out.close();
//System.out.println("压缩成功!目标文件名为:"+targetFileName);
flag=true;
}catch(FileNotFoundExceptione){
e.printStackTrace();
}catch(IOExceptione){
e.printStackTrace();
}
returnflag;
}

/**
*将文件压缩到Zip输出流
*
*@parambaseDirPath
*根目录路径
*@paramfile
*要压缩的文件
*@paramout
*输出流
*@throwsIOException
*/
privatestaticvoidfileToZip(StringbaseDirPath,Filefile,
ZipOutputStreamout)throwsIOException{
//
FileInputStreamin=null;
org.apache.tools.zip.ZipEntryentry=null;
//创建复制缓冲区1024*4=4K
byte[]buffer=newbyte[1024*4];
intbytes_read=0;
if(file.isFile()){
in=newFileInputStream(file);
//根据parent路径名字符串和child路径名字符串创建一个新File实例
StringzipFileName=getEntryName(baseDirPath,file);
entry=neworg.apache.tools.zip.ZipEntry(zipFileName);
//"压缩文件"对象加入"要压缩的文件"对象
out.putNextEntry(entry);
//现在是把"要压缩的文件"对象中的内容写入到"压缩文件"对象
while((bytes_read=in.read(buffer))!=-1){
out.write(buffer,0,bytes_read);
}
out.closeEntry();
in.close();
//System.out.println("添加文件"+file.getAbsolutePath()+"被添加到ZIP
//文件中!");
}
}

/**
*多个文件目录压缩到Zip输出流
*
*@parambaseDirPath
*@paramfiles
*@paramout
*@throwsIOException
*/
@SuppressWarnings("unused")
privatestaticvoidfilesToZip(StringbaseDirPath,File[]files,
ZipOutputStreamout)throwsIOException{
//遍历所有的文件一个一个地压缩
for(inti=0;i<files.length;i++){
Filefile=files[i];
if(file.isFile()){
//调用本类的一个静态方法压缩一个文件
AntZipUtil.fileToZip(baseDirPath,file,out);
}else{
/*
*这是一个文件夹所以要再次得到它下面的所有的文件这里是自己调用自己..............递归..........
*/
AntZipUtil.dirToZip(baseDirPath,file,out);
}
}
}

/**
*将文件目录压缩到Zip输出流
*
*@parambaseDirPath
*@paramdir
*@paramout
*@throwsIOException
*/
privatestaticvoiddirToZip(StringbaseDirPath,Filedir,
ZipOutputStreamout)throwsIOException{
//得到一个文件列表(本目录下的所有文件对象集合)
File[]files=dir.listFiles();
//要是这个文件集合数组的长度为0,也就证明了这是一个空的文件夹,虽然没有再循环遍历它的必要,但是也要把这个空文件夹也压缩到目标文件中去
if(files.length==0){
//根据parent路径名字符串和child路径名字符串创建一个新File实例
StringzipFileName=getEntryName(baseDirPath,dir);
org.apache.tools.zip.ZipEntryentry=neworg.apache.tools.zip.ZipEntry(
zipFileName);
out.putNextEntry(entry);
out.closeEntry();
}else{
//遍历所有的文件一个一个地压缩
for(inti=0;i<files.length;i++){
Filefile=files[i];
if(file.isFile()){
//调用本类的一个静态方法压缩一个文件
AntZipUtil.fileToZip(baseDirPath,file,out);
}else{
/*
*这是一个文件夹所以要再次得到它下面的所有的文件
*这里是自己调用自己..............递归..........
*/
AntZipUtil.dirToZip(baseDirPath,file,out);
}
}
}
}

/**
*获取待压缩文件在ZIP文件中的entry的名字,即相对于根目录的相对路径名
*
*@parambaseDirPath
*根目录
*@paramfile
*@return
*/
privatestaticStringgetEntryName(StringbaseDirPath,Filefile){
/**
*改变baseDirPath的形式把"C:/temp"变成"C:/temp/"
*/
if(!baseDirPath.endsWith(File.separator)){
baseDirPath+=File.separator;
}
StringfilePath=file.getAbsolutePath();
/**
*测试此抽象路径名表示的文件是否是一个目录。要是这个文件对象是一个目录则也要变成后面带"/"这个文件对象类似于
*"C:/temp/人体写真/1.jpg"要是这个文件是一个文件夹则也要变成后面带"/"
*因为你要是不这样做,它也会被压缩到目标文件中但是却不能正解显示也就是说操作系统不能正确识别它的文件类型(是文件还是文件夹)
*/
if(file.isDirectory()){
filePath+="/";
}
intindex=filePath.indexOf(baseDirPath);
returnfilePath.substring(index+baseDirPath.length());
}

////////////////////////////解压缩////////////////////////////////////////
/**
*调用org.apache.tools.zip实现解压缩,支持目录嵌套和中文名
*也可以使用java.util.zip不过如果是中文的话,解压缩的时候文件名字会是乱码。原因是解压缩软件的编码格式跟java.util.zip.ZipInputStream的编码字符集(固定是UTF-8)不同
*
*@paramzipFileName
*要解压缩的文件
*@paramoutputDirectory
*要解压到的目录
*@throwsException
*/
publicstaticbooleanunZip(StringzipFileName,StringoutputDirectory){
booleanflag=false;
try{
org.apache.tools.zip.ZipFilezipFile=neworg.apache.tools.zip.ZipFile(
zipFileName);
java.util.Enumeratione=zipFile.getEntries();
org.apache.tools.zip.ZipEntryzipEntry=null;
createDirectory(outputDirectory,"");
while(e.hasMoreElements()){
zipEntry=(org.apache.tools.zip.ZipEntry)e.nextElement();
//System.out.println("unziping"+zipEntry.getName());
if(zipEntry.isDirectory()){
Stringname=zipEntry.getName();
name=name.substring(0,name.length()-1);
Filef=newFile(outputDirectory+File.separator+name);
f.mkdir();
System.out.println("创建目录:"+outputDirectory
+File.separator+name);
}else{
StringfileName=zipEntry.getName();
fileName=fileName.replace('\\','/');
//System.out.println("测试文件1:"+fileName);
if(fileName.indexOf("/")!=-1){
createDirectory(outputDirectory,fileName.substring(0,
fileName.lastIndexOf("/")));
fileName=fileName.substring(
fileName.lastIndexOf("/")+1,fileName
.length());
}

Filef=newFile(outputDirectory+File.separator
+zipEntry.getName());

f.createNewFile();
InputStreamin=zipFile.getInputStream(zipEntry);
FileOutputStreamout=newFileOutputStream(f);

byte[]by=newbyte[1024];
intc;
while((c=in.read(by))!=-1){
out.write(by,0,c);
}
out.close();
in.close();
}
flag=true;
}
}catch(Exceptionex){
ex.printStackTrace();
}
returnflag;
}

/**
*创建目录
*
*@paramdirectory
*父目录
*@paramsubDirectory
*子目录
*/
privatestaticvoidcreateDirectory(Stringdirectory,StringsubDirectory){
Stringdir[];
Filefl=newFile(directory);
try{
if(subDirectory==""&&fl.exists()!=true)
fl.mkdir();
elseif(subDirectory!=""){
dir=subDirectory.replace('\\','/').split("/");
for(inti=0;i<dir.length;i++){
FilesubFile=newFile(directory+File.separator+dir[i]);
if(subFile.exists()==false)
subFile.mkdir();
directory+=File.separator+dir[i];
}
}
}catch(Exceptionex){
System.out.println(ex.getMessage());
}
}

///////////////////////////////////////

publicstaticvoidmain(String[]temp){
//压缩
StringbaseDirName="C:\\";
String[]fileNames={"中文1.doc","中文2.doc"};
StringzipFileName="c:\\中文.zip";
//压缩多个指定的文件到ZIP
System.out.println(AntZipUtil.zip(baseDirName,fileNames,zipFileName,"GBK"));

//压缩一个文件夹到ZIP
StringsourcePath="c:\\test\\";
StringzipFilePath="c:\\中文2.zip";
AntZipUtil.zip(sourcePath,zipFilePath);

//解压缩
//System.out.println(AntZipUtil.unZip("c:\\中文.zip","c:\\中文"));
}

}

package com.cliff.common; import java.io.File; import java.io.FileInputStream; import java.io.FileNotFoundException; import java.io.FileOutputStream; import java.io.IOException; import java.io.InputStream; import java.util.Enumeration; import org.apache.tools.zip.ZipEntry; import org.apache.tools.zip.ZipFile; import org.apache.tools.zip.ZipOutputStream; /** * * 类名: ZipUtil.java * 描述:压缩/解压缩zip处理类 * 创建者:XXX * 创建日期:2015年5月7日 - 下午1:35:02 * 版本: V0.1 * 修改者: * 修改日期: */ public class ZipUtil { /** * * 功能描述:压缩文件 * 创建者:XXX * 创建日期: 2015年5月7日 - 下午1:35:18 * 版本: V0.1 * 修改者: * 修改日期: * @param directory 指定压缩文件路径 压缩到同目录 * @throws IOException * void */ public static void zip(String directory) throws FileNotFoundException, IOException { zip("", null, directory); } /** * * 功能描述:压缩文件 * 创建者:XXX * 创建日期: 2015年5月7日 - 下午1:36:03 * 版本: V0.1 * 修改者: * 修改日期: * @param zipFileName 压缩产生的zip文件名--带路径,如果为null或空则默认按文件名生产压缩文件名 * @param relativePath 相对路径,默认为空 * @param directory 文件或目录的绝对路径 * void */ public static void zip(String zipFileName, String relativePath, String directory) throws FileNotFoundException, IOException { String fileName = zipFileName; if (fileName == null || fileName.trim().equals("")) { File temp = new File(directory); if (temp.isDirectory()) { fileName = directory + ".zip"; } else { if (directory.indexOf(".") > 0) { fileName = directory.substring(0, directory.lastIndexOf("."))+ "zip"; } else { fileName = directory + ".zip"; } } } ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(fileName)); try { zip(zos, relativePath, directory); } catch (IOException ex) { throw ex; } finally { if (null != zos) { zos.close(); } } } /** * * 功能描述:压缩文件 * 创建者:XXX * 创建日期: 2015年5月7日 - 下午1:37:55 * 版本: V0.1 * 修改者: * 修改日期: * @param zos 压缩输出流 * @param relativePath 相对路径 * @param absolutPath 文件或文件夹绝对路径 * @throws IOException * void */ private static void zip(ZipOutputStream zos, String relativePath, String absolutPath) throws IOException { File file = new File(absolutPath); if (file.isDirectory()) { File[] files = file.listFiles(); for (int i = 0; i < files.length; i++) { File tempFile = files[i]; if (tempFile.isDirectory()) { String newRelativePath = relativePath + tempFile.getName() + File.separator; createZipNode(zos, newRelativePath); zip(zos, newRelativePath, tempFile.getPath()); } else { zipFile(zos, tempFile, relativePath); } } } else { zipFile(zos, file, relativePath); } } /** * * 功能描述:压缩文件 * 创建者:XXX * 创建日期: 2015年5月7日 - 下午1:38:46 * 版本: V0.1 * 修改者: * 修改日期: * @param zos 压缩输出流 * @param file 文件对象 * @param relativePath 相对路径 * @throws IOException * void */ private static void zipFile(ZipOutputStream zos, File file, String relativePath) throws IOException { ZipEntry entry = new ZipEntry(relativePath + file.getName()); zos.putNextEntry(entry); InputStream is = null; try { is = new FileInputStream(file); int BUFFERSIZE = 2 <= 0) { zos.write(buffer, 0, length); } zos.flush(); zos.closeEntry(); } catch (IOException ex) { throw ex; } finally { if (null != is) { is.close(); } } } /** * * 功能描述:创建目录 * 创建者:XXX * 创建日期: 2015年5月7日 - 下午1:39:12 * 版本: V0.1 * 修改者: * 修改日期: * @param zos zip输出流 * @param relativePath 相对路径 * @throws IOException * void */ private static void createZipNode(ZipOutputStream zos, String relativePath) throws IOException { ZipEntry zipEntry = new ZipEntry(relativePath); zos.putNextEntry(zipEntry); zos.closeEntry(); } /** * * 功能描述:解压缩文件 * 创建者:XXX * 创建日期: 2015年5月7日 - 下午1:39:32 * 版本: V0.1 * 修改者: * 修改日期: * @param zipFilePath zip文件路径 * @param targetPath 解压缩到的位置,如果为null或空字符串则默认解压缩到跟zip同目录跟zip同名的文件夹下 * void */ public static void unzip(String zipFilePath, String targetPath) throws IOException { InputStream is = null; FileOutputStream fileOut = null; File file = null; ZipFile zipFile = null; try { zipFile = new ZipFile(zipFilePath,"GBK"); String directoryPath = ""; if (null == targetPath || "".equals(targetPath)) { directoryPath = zipFilePath.substring(0, zipFilePath.lastIndexOf(".")); } else { directoryPath = targetPath; } for(Enumeration entries = zipFile.getEntries(); entries.hasMoreElements();){ ZipEntry entry = (ZipEntry)entries.nextElement(); file = new File(directoryPath+"/"+entry.getName()); if(entry.isDirectory()){ file.mkdirs(); }else{ //如果指定文件的目录不存在,则创建之. File parent = file.getParentFile(); if(!parent.exists()){ parent.mkdirs(); } is = zipFile.getInputStream(entry); fileOut = new FileOutputStream(file); int readLen = 0; byte[] buffer = new byte[4096]; while ((readLen = is.read(buffer, 0, 4096)) >= 0) { fileOut.write(buffer, 0, readLen); } fileOut.close(); is.close(); } } zipFile.close(); } catch (IOException ex) { throw ex; } finally { if(null != zipFile){ zipFile = null; } if (null != is) { is.close(); } if (null != fileOut) { fileOut.close(); } } } /** * * 功能描述:生产文件 如果文件所在路径不存在则生成路径 * 创建者:XXX * 创建日期: 2015年5月7日 - 下午1:41:04 * 版本: V0.1 * 修改者: * 修改日期: * @param fileName 文件名 带路径 * @param isDirectory 是否为路径 * @return * File */ public static File buildFile(String fileName, boolean isDirectory) { File target = new File(fileName); if (isDirectory){ target.mkdirs(); } else { if (!target.getParentFile().exists()) { target.getParentFile().mkdirs(); target = new File(target.getAbsolutePath()); } } return target; } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值