Java文件压缩

import java.io.File;  
import java.io.FileInputStream;  
import java.io.FileOutputStream;  
import java.io.InputStream;  
import java.util.Enumeration;  
import java.util.zip.ZipEntry;  
import java.util.zip.ZipFile;  
import java.util.zip.ZipOutputStream;  
  
//错误小结:  
//1 关于file.isFile()与file.isDirectory()记忆出了偏差.  
//  错误以为若是Directory一定是file,file不一定是Directory  
//  更正:file和Directory是两码事.只能是file或Directory中其中一个.  
  
//2 并不是说我们执行一句File f=new File("F:\\x.txt");  
//  在本地硬盘上就生成了一个x.txt文件.而应该进行如下的  
//  操作才可以.  
//  File f=new File("F:\\x.txt");  
//  if (!f.exists()) {  
//     f.createNewFile();  
//  }  
// 其中f.createNewFile()表示创建了一个空文件  
//  
// 多数的情况下,我们执行了File f=new File("F:\\x.txt")  
// 以后再利用输入流,输出流对f进行操作,比如往该x.txt文件中  
// 写入hello world  
//   
  
//3 血的教训:  
//  zip()方法中没有关闭流zos.导致压缩后的文件解压时出错  
  
//重要总结:  
//1 关于zip和unzip的核心操作对象是每一个文件!!!  
//  比如碰到一个directory,那么会去遍历里面的每一个文件,挨个对其进行压缩.  
//  不要错误地理解,若是一个directory,那么会将其作为一个整体进行压缩.  
  
//2 在JAVA中每一个压缩文件都用一个ZipEntry表示  
//  所以在压缩的过程中要得到每个文件的完整路径(从最外层文件夹至文件本身)  
//  用此完整路径为每个压缩文件new()一个ZipEntry  
  
//3 所以利用zipEntry可记录原来的目录层次.解压后才保持原样  
//  也可以看到在解压时利用entrys.hasMoreElements()来挨个  
//  解压每个zipEntry.  
//  参见解压中的代码:  
//  perUnzipFilePath = unzipPath + zipEntry.getName();  
      
public class TestZipAndUnZip {  
   public static void main(String[] args) throws Exception {  
       TestZipAndUnZip test=new TestZipAndUnZip();  
         
       //压缩和解压单个文件  
       test.zip2("F:\\kk\\cc.txt","F:\\11.zip");  
       test.unZipFile2("F:\\11.zip", "F:\\test11");  
         
       //压缩和解压一个目录  
       test.zip2("F:\\kk","F:\\22.zip");  
       test.unZipFile2("F:\\22.zip", "F:\\test22");  
         
   }  
      
    /** 
     * 该方法将一个给定路径的文件压缩 
     * @param willZipPath 待压缩文件的路径 
     * @param zipedPath   该文件压缩后的路径 
     */  
    public void zip2(String willZipPath, String zipedPath) {  
        try {  
            File willZipFile = new File(willZipPath);  
            File zipedFile = new File(zipedPath);  
            ZipOutputStream zos = new ZipOutputStream(new FileOutputStream(zipedFile));  
            if (willZipFile.isFile()) {  
                fileToZip2(willZipPath, zos);  
            }  
            if (willZipFile.isDirectory()) {  
                dirToZip2(willZipPath, willZipFile, zos);  
            }  
            // 方法调用完成后关闭流  
            zos.close();  
        } catch (Exception e) {  
        }  
    }  
      
      
    /** 
     * @param willZipFilePath  待压缩文件的路径 
     * @param zos              压缩文件输出流 
     * 1 关于以下两句代码 
     * ZipEntry entry = new ZipEntry(); 
     * zos.putNextEntry(entry); 
     * 把生成的ZipEntry对象加入到压缩文件中 
     * 之后往压缩文件中写入的内容均放在该ZipEntry对象中 
     *  
     * 2 fis.close()但是不能在此处zos.close() 
     * 因为该zos是上一方法传递过来的.可能在压缩目录的时候会 
     * 再次使用到该zos流.若此时关闭,则导致目录中的一个文件 
     * 被压缩 
     */  
    public void fileToZip2(String willZipFilePath,ZipOutputStream zos){  
        try {  
            File willZipFile=new File(willZipFilePath);  
            ZipEntry entry = new ZipEntry(getEntryName2(willZipFilePath, willZipFile));  
            zos.putNextEntry(entry);  
            FileInputStream fis = new FileInputStream(willZipFilePath);  
            int len = 0;  
            while ((len = fis.read()) != -1){  
                zos.write(len);  
            }  
            fis.close();  
            //流关闭错误!  
            //zos.close();  
        } catch (Exception e) {  
        }  
    }  
      
    /** 
     * @param willZipDirctoryPath   原目录所在路径 
     * @param willZipedDirectory    原目录 
     * @param zos                   压缩流 
     * 注意: 
     * 在处理空文件夹的时候 
     * getEntryName2(willZipDirctoryPath, willZipedDirectory)+"/" 
     * 中的"/"是必不可少的 
     */  
    public void dirToZip2(String willZipDirctoryPath,File willZipedDirectory, ZipOutputStream zos) {  
            if (willZipedDirectory.isDirectory()) {  
                File[] files = willZipedDirectory.listFiles();  
                //处理空文件夹的情况  
                if (files.length==0) {  
                    ZipEntry zipEntry=new ZipEntry  
                    (getEntryName2(willZipDirctoryPath, willZipedDirectory)+"/");  
                    try {  
                        zos.putNextEntry(zipEntry);  
                    } catch (Exception e) {  
                        e.printStackTrace();  
                    }  
                    return;  
                }  
                for (int i = 0; i < files.length; i++) {  
                    File file = files[i];  
                    //若是文件,递归调用fileToZip()  
                    if (file.isFile()) {  
                        fileToZip2(file.getAbsolutePath(), zos);  
                    }  
                    //若是文件,递归调用dirToZip()  
                    if (file.isDirectory()) {  
                        dirToZip2(file.getAbsolutePath(),file, zos);  
                    }  
                }  
            }  
        }  
      
      
    /** 
     * @param rawPath  需要压缩的目录或者文件的完整路径 
     * @param file     需要压缩的文件或者目录 
     * @return         entryName 
     *  
     * 该方法返回EntryName,表示从最外层目录开始到该文件(目录) 
     * 的完整路径 
     * 备注: 
     * 1 该示例中文件均存放在某盘下,如E:\所以rawPath.substring(3); 
     * 2 注释中"@param file 需要压缩的文件或者目录".其实绝大多数情况下 
     *  都是文件,只有一种情况是目录,就是空文件夹的情况.  
     */  
    public String getEntryName2(String rawPath,File file){  
        try {  
            String rawDir=rawPath.substring(3);  
            int rawDirIndex=file.getAbsolutePath().indexOf(rawDir);  
            String entryName=file.getAbsolutePath().substring(rawDirIndex);  
            return entryName;  
        } catch (Exception e) {  
        }  
        return null;  
    }  
  
      
    /** 
     * @param zipedFilePath  原压缩文件的路径 
     * @param unzipPath      文件解压后的路径 
     * 对于文件或者目录操作的小梳理: 
     * 1 对于目录应该先执行file.mkdir(s)()才可以 
     *  往里面其下存入文件.比如: 
     *  File f=new File("F:\\test\\x.txt"); 
        if (!f.exists()) { 
                f.createNewFile(); 
        } 
        这当然要报错,因为x.txt的所在目录还不存在!! 
        所以应该改正为: 
        File f=new File("F:\\test\\x.txt"); 
        f.getParentFile().mkdirs(); 
        if (!f.exists()) { 
                f.createNewFile(); 
        } 
        2同样的道理 
        File f=new File("F:\\test\\x.txt"); 
        if (f.isFile()) { 
            System.out.println("true"); 
        }else{ 
            System.out.println("false"); 
        } 
        结果为false 
        3类似的问题 
        File f=new File("F:\\x.txt"); 
        if (f.isFile()) { 
            System.out.println("true"); 
        }else{ 
            System.out.println("false"); 
        } 
        结果为false 
        因为只是new了一个File,并没有创建!!! 
        File f=new File("F:\\x.txt"); 
        f.createNewFile(); 
        if (f.isFile()) { 
            System.out.println("true"); 
        }else{ 
            System.out.println("false"); 
        } 
        此时为true 
         
        此处: 
        if (zipEntry.isDirectory()) { 
            new File(perUnzipFilePath).mkdirs(); 
        } else { 
            new File(perUnzipFilePath).getParentFile().mkdirs(); 
        } 
        已经建立了每个文件夹. 
        然后才开始对每个空文件夹和每个文件进行流操作. 
        和上面的道理一致,刚开始犯了个错误,没有采用 
        else { 
                fos = new FileOutputStream(perUnzipFile); 
                is = zipFile.getInputStream(zipEntry); 
                while ((len = is.read(buffer)) != -1) { 
                    fos.write(buffer, 0, len); 
                } 
            } 
        而是用if(perUnzipFile.isFile()){}来操作,这当然错了. 
        因为该perUnzipFile没有执行perUnzipFile.createNewFile(); 
        所以它还不是File.类似的情况,多采用流操作来进行读写. 
        所以总结两种与File有关的操作: 
        1 File f=new File(""); 
          f.createNewFile(); 
             然后对f操作 
        2 file f=new File(""); 
            然后用输入输出流进行流操作 
            举例子: 
          File f=new File("F:\\2221x.txt"); 
          FileOutputStream fos=new FileOutputStream(f); 
          String string="hello"; 
          byte []b=string.getBytes(); 
          fos.write(b, 0, b.length);         
             该例子是正确的. 
             疑问:没有执行 f.createNewFile()为什么不报错.因为输出流 
          FileOutputStream已经帮我们做了该工作了. 
            修改例子即可知: 
            File f=new File("F:\\2221x.txt"); 
            if (f.isFile()) { 
                System.out.println("true1"); 
            } else { 
                System.out.println("false1"); 
            } 
            FileOutputStream fos=new FileOutputStream(f); 
            if (f.isFile()) { 
                System.out.println("true2"); 
            } else { 
                System.out.println("false2"); 
            } 
            String string="hello"; 
            byte []b=string.getBytes(); 
            fos.write(b, 0, b.length); 
            输出false1,true2 
            这就得到了验证. 
         
     */  
    public void unZipFile2(String zipedFilePath, String unzipPath) {  
        FileOutputStream fos=null;  
        InputStream is=null;  
        ZipEntry zipEntry=null;  
        String perUnzipFilePath=null;  
        if (!unzipPath.endsWith(File.separator)) {  
            unzipPath+=File.separator;  
        }  
        try {  
            ZipFile zipFile=new ZipFile(zipedFilePath);  
            Enumeration entries=zipFile.entries();  
            byte [] buffer=new byte[1024*8];  
            int len=0;  
            while (entries.hasMoreElements()) {  
                zipEntry = (ZipEntry) entries.nextElement();  
                perUnzipFilePath = unzipPath + zipEntry.getName();  
                //1建立每个文件夹  
                if (zipEntry.isDirectory()) {  
                    //处理空文件夹的情况  
                    //创建空目录  
                    new File(perUnzipFilePath).mkdirs();  
                } else {  
                    //为每个文件创立其所在目录  
                    new File(perUnzipFilePath).getParentFile().mkdirs();  
                }  
                //2在用流操作处理每个文件夹中的文件  
                // 2.1if中的操作只是针对空目录而进行的.所以if中的代码可以  
                // 注释掉,无实际意义.  
                // 2.2else中的操作是对于每个具体的文件而进行的流操作  
                File perUnzipFile = new File(perUnzipFilePath);  
                if (perUnzipFile.isDirectory()) {  
                    File[] files = perUnzipFile.listFiles();  
                    for (int i = 0; i < files.length; i++) {  
                        File file = files[i];  
                        fos = new FileOutputStream(file);  
                        is = zipFile.getInputStream(zipEntry);  
                        while ((len = is.read(buffer)) != -1) {  
                            fos.write(buffer, 0, len);  
                        }  
                    }  
                } else {  
                    fos = new FileOutputStream(perUnzipFile);  
                    is = zipFile.getInputStream(zipEntry);  
                    while ((len = is.read(buffer)) != -1) {  
                        fos.write(buffer, 0, len);  
                    }  
                }  
            }  
            if (fos!=null) {  
                fos.close();  
            }  
            if (is!=null) {  
                is.close();  
            }  
        } catch (Exception e) {  
            e.printStackTrace();  
        }  
    }  
  
}   
谷哥的小弟 http://blog.youkuaiyun.com/lfdfhl
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

编程之艺术

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值