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

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

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

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
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.ArrayList;
import java.util.List;
import java.util.zip.CRC32;
import java.util.zip.CheckedOutputStream;
import java.util.zip.Deflater;

import org.apache.tools.zip.ZipEntry;
import org.apache.tools.zip.ZipOutputStream;

/**
 * 功能:使用Apache Ant里提供的org.apache.tools.zip实现zip压缩和解压 (支持中文文件名)
 * 解决了由于java.util.zip包不支持汉字的问题。 使用java.util.zip包时,当zip文件中有名字为中文的文件时,
 * 就会出现异常:"Exception in thread "main " java.lang.IllegalArgumentException at
 * java.util.zip.ZipInputStream.getUTF8String(ZipInputStream.java:285)
 * 
 * @author 夏明龙 E-mail:邮箱
 * @version 创建时间:2013-3-22 上午10:40:21 类说明:
 */
public class AntZipUtil {
 private static List list = new ArrayList();

 private static List listFile(String path) {
  File file = new File(path);
  String[] array = null;
  String sTemp = "";

  if (!file.isDirectory()) {
   return null;
  }
  array = file.list();
  if (array.length > 0) {
   for (int i = 0; i < array.length; i++) {
    sTemp = path + array[i];
    file = new File(sTemp);
    if (file.isDirectory()) {
     listFile(sTemp + "/");
    } else
     list.add(sTemp);
   }
  } else {
   return null;
  }

  return list;
 }

 public static void zip(String needtozipfilepath, String zipfilepath){
  try {
   byte[] b = new byte[512];

   File needtozipfile = new File(needtozipfilepath);

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

   String zipFile = zipfilepath;
   File targetFile = new File(zipFile.substring(0, zipFile.indexOf("\\") + 1));

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

   String filepath = needtozipfilepath;
   List fileList = listFile(filepath);
   FileOutputStream fileOutputStream = new FileOutputStream(zipFile);
   CheckedOutputStream cs = new CheckedOutputStream(fileOutputStream,new CRC32());
   ZipOutputStream out = new ZipOutputStream(new BufferedOutputStream(cs));

   for (int i = 0; i < fileList.size(); i++) {
    InputStream in = new FileInputStream((String) fileList.get(i));
    String fileName = ((String) fileList.get(i)).replace(File.separatorChar, '/');
    fileName = fileName.substring(fileName.indexOf("/") + 1);
    ZipEntry e = new ZipEntry(fileName);
    out.putNextEntry(e);
    int len = 0;
    while ((len = in.read(b)) != -1) {
     out.write(b, 0, len);
    }
    out.closeEntry();
   }
   out.close();
  } catch (Exception e) {
   e.printStackTrace();
  }
 }

 // ///
 /**
  * 压缩文件 或者 文件夹
  * 
  * @param baseDirName
  *            压缩的根目录
  * @param fileName
  *            根目录下待压缩的文件或文件夹名
  * @param targetFileName
  *            目标ZIP 文件 星号 "*" 表示压缩根目录下的全部文件
  * 
  */
 public static boolean zip(String baseDirName, String[] fileNames,
   String targetFileName, String encoding) {
  boolean flag = false;
  try {
   // 判断 "压缩的根目录"是否存在! 是否是一个文件夹!
   File baseDir = new File(baseDirName);
   if (!baseDir.exists() || (!baseDir.isDirectory())) {
    System.err.println("压缩失败! 根目录不存在: " + baseDirName);
    return false;
   }

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

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

   // 设置压缩编码Apache Ant有个包专门处理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 = new File[fileNames.length];
    for (int i = 0; i < files.length; i++) {
     // 根据 parent 抽象路径名和 child 路径名字符串创建一个新 File 实例。
     files[i] = new File(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 (FileNotFoundException e) {
   e.printStackTrace();
  } catch (IOException e) {
   e.printStackTrace();
  }
  return flag;
 }

 /**
  * 将文件压缩到Zip 输出流
  * 
  * @param baseDirPath
  *            根目录路径
  * @param file
  *            要压缩的文件
  * @param out
  *            输出流
  * @throws IOException
  */
 private static void fileToZip(String baseDirPath, File file,
   ZipOutputStream out) throws IOException {
  //
  FileInputStream in = null;
  org.apache.tools.zip.ZipEntry entry = null;
  // 创建复制缓冲区 1024*4 = 4K
  byte[] buffer = new byte[1024 * 4];
  int bytes_read = 0;
  if (file.isFile()) {
   in = new FileInputStream(file);
   // 根据 parent 路径名字符串和 child 路径名字符串创建一个新 File 实例
   String zipFileName = getEntryName(baseDirPath, file);
   entry = new org.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 输出流
  * 
  * @param baseDirPath
  * @param files
  * @param out
  * @throws IOException
  */
 @SuppressWarnings("unused")
 private static void filesToZip(String baseDirPath, File[] files,
   ZipOutputStream out) throws IOException {
  // 遍历所有的文件 一个一个地压缩
  for (int i = 0; i < files.length; i++) {
   File file = files[i];
   if (file.isFile()) {
    // 调用本类的一个静态方法 压缩一个文件
    AntZipUtil.fileToZip(baseDirPath, file, out);
   } else {
    /*
     * 这是一个文件夹 所以要再次得到它下面的所有的文件 这里是自己调用自己..............递归..........
     */
    AntZipUtil.dirToZip(baseDirPath, file, out);
   }
  }
 }

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

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

 // //解压缩
 /**
  * 调用org.apache.tools.zip实现解压缩,支持目录嵌套和中文名
  * 也可以使用java.util.zip不过如果是中文的话,解压缩的时候文件名字会是乱码。原因是解压缩软件的编码格式跟java.util.zip.ZipInputStream的编码字符集(固定是UTF-8)不同
  * 
  * @param zipFileName
  *            要解压缩的文件
  * @param outputDirectory
  *            要解压到的目录
  * @throws Exception
  */
 public static boolean unZip(String zipFileName, String outputDirectory) {
  boolean flag = false;
  try {
   org.apache.tools.zip.ZipFile zipFile = new org.apache.tools.zip.ZipFile(
     zipFileName);
   java.util.Enumeration e = zipFile.getEntries();
   org.apache.tools.zip.ZipEntry zipEntry = null;
   createDirectory(outputDirectory, "");
   while (e.hasMoreElements()) {
    zipEntry = (org.apache.tools.zip.ZipEntry) e.nextElement();
    // System.out.println("unziping " + zipEntry.getName());
    if (zipEntry.isDirectory()) {
     String name = zipEntry.getName();
     name = name.substring(0, name.length() - 1);
     File f = new File(outputDirectory + File.separator + name);
     f.mkdir();
     System.out.println("创建目录:" + outputDirectory
       + File.separator + name);
    } else {
     String fileName = 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());
     }

     File f = new File(outputDirectory + File.separator
       + zipEntry.getName());

     f.createNewFile();
     InputStream in = zipFile.getInputStream(zipEntry);
     FileOutputStream out = new FileOutputStream(f);

     byte[] by = new byte[1024];
     int c;
     while ((c = in.read(by)) != -1) {
      out.write(by, 0, c);
     }
     out.close();
     in.close();
    }
    flag = true;
   }
  } catch (Exception ex) {
   ex.printStackTrace();
  }
  return flag;
 }

 /**
  * 创建目录
  * 
  * @param directory
  *            父目录
  * @param subDirectory
  *            子目录
  */
 private static void createDirectory(String directory, String subDirectory) {
  String dir[];
  File fl = new File(directory);
  try {
   if (subDirectory == "" && fl.exists() != true)
    fl.mkdir();
   else if (subDirectory != "") {
    dir = subDirectory.replace('\\', '/').split("/");
    for (int i = 0; i < dir.length; i++) {
     File subFile = new File(directory + File.separator + dir[i]);
     if (subFile.exists() == false)
      subFile.mkdir();
     directory += File.separator + dir[i];
    }
   }
  } catch (Exception ex) {
   System.out.println(ex.getMessage());
  }
 }

 // /

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

  //压缩一个文件夹 到ZIP
  String sourcePath = "c:\\test\\";
  String zipFilePath = "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、付费专栏及课程。

余额充值