zoukankan      html  css  js  c++  java
  • Java Zip压缩实现

    最近在自学javaWeb,先复习一下java,把还给老师的东西再找回来(知识如果不用很快就会忘记啊)。。

    今天看到了zip压缩,决定要整理一下。

    java将有关zip压缩的内容都封装在java.util.zip宝中,用java实现zip压缩,不用考虑压缩算法,java已经将这些进行了封装

    实际上用java实现zip压缩涉及的就是一个“输入输出流”的概念

    用java实现一个文件的zip压缩,过程可以简单地表示为:

    当然具体实现要比这个复杂一点,比如要先像zip文件中写入目录进入点。。如果要压缩文件夹中的内容要遍历文件夹中的文件和子文件夹。

    先给出源代码,再解释:

    import java.io.*;
    import java.util.*;
    import java.util.zip.*;
    
    class ZipCompress
    {
        private String zipFileName;      // 目的地Zip文件
        private String sourceFileName;   //源文件(带压缩的文件或文件夹)
        
        public ZipCompress(String zipFileName,String sourceFileName)
        {
            this.zipFileName=zipFileName;
            this.sourceFileName=sourceFileName;
        }
        
        public void zip() throws Exception
        {
            //File zipFile = new File(zipFileName);
            System.out.println("压缩中...");
            
            //创建zip输出流
            ZipOutputStream out = new ZipOutputStream( new FileOutputStream(zipFileName));
            
            //创建缓冲输出流
            BufferedOutputStream bos = new BufferedOutputStream(out);
            
            File sourceFile = new File(sourceFileName);
            
            //调用函数
            compress(out,bos,sourceFile,sourceFile.getName());
            
            bos.close();
            out.close();
            System.out.println("压缩完成");
            
        }
        
        public void compress(ZipOutputStream out,BufferedOutputStream bos,File sourceFile,String base) throws Exception
        {
            //如果路径为目录(文件夹)
            if(sourceFile.isDirectory())
            {
            
                //取出文件夹中的文件(或子文件夹)
                File[] flist = sourceFile.listFiles();
                
                if(flist.length==0)//如果文件夹为空,则只需在目的地zip文件中写入一个目录进入点
                {
                    System.out.println(base+"/");
                    out.putNextEntry(  new ZipEntry(base+"/") );
                }
                else//如果文件夹不为空,则递归调用compress,文件夹中的每一个文件(或文件夹)进行压缩
                {
                    for(int i=0;i<flist.length;i++)
                    {
                        compress(out,bos,flist[i],base+"/"+flist[i].getName());
                    }
                }
            }
            else//如果不是目录(文件夹),即为文件,则先写入目录进入点,之后将文件写入zip文件中
            {
                out.putNextEntry( new ZipEntry(base) );
                FileInputStream fos = new FileInputStream(sourceFile);
                BufferedInputStream bis = new BufferedInputStream(fos);
                
                int tag;
                System.out.println(base);
                //将源文件写入到zip文件中
                while((tag=bis.read())!=-1)
                {
                    bos.write(tag);
                }
                bis.close();
                fos.close();
                
            }
        }
    }
    public class TestZip 
    {
    
        public static void main(String[] args)
        {
            ZipCompress zipCom = new ZipCompress("D:\电影.zip","F:\电影");
            try
            {
                zipCom.zip();
            }
            catch(Exception e)
            {
                e.printStackTrace();
            }
        }
    }

    zip()方法中首先创建了ZipOutputStream(zip输出流)之后调用compress()方法

    在compress()方法中通过递归完成将文件夹中的每个文件的压缩。

    具体为:

    首先判断输入的文件是文件夹还是文件,如果是文件,直接压缩,如果是文件夹则要递归调用compress()遍历压缩文件夹中的文件。

    如果是文件,首先在zip文件中写入一个目录进入点(所谓目录进入点,个人理解就是源文件在目标zip文件中的路径

    然后用文件输入流FileInputStream将源文件读入内存,在用ZipOutputStream(zip输出流)将读入内存的数据写入到zip文件中,这就完成了压缩。(实际上就是输入输出流

    当文件夹为空的时候,只需写入一个目录进入点(要注意最后一定要加一个"",表示一个目录)。

    运行结果为:

    源文件:

    其中文件夹1中含有两个文件,文件夹2 为空。

    压缩后的文件为:

    前面提到的空文件夹的目录进入点一定要加"",以表示一个文件夹,而不是一个文件,我们可以把程序中的""去掉,这时结果为:

    可以看到本来是文件夹的文件夹2,编程了文件。所以可以把目录进入点就理解为,源文件在zip文件中对应的路径,这个相对路径应该与源文件夹中的相对路径一致,当然也可以不一致,比如我们在base前加入一个字符串"我爱",

    out.putNextEntry( new ZipEntry("我爱"+base) );

    结果为:

    可以看到zip中的根目录变成了“我爱电影”而不是原来的电影

     水平有限。。。请多多指教。。。。

  • 相关阅读:
    构造函数和属性初始化
    C#3.0新增功能06 对象和集合初始值设定项
    C#动态操作DataTable(新增行、列、查询行、列等)
    快速排序
    HTML5原生拖放实例分析
    从web移动端布局到react native布局
    Chrome浏览器Network面板http请求时间分析
    CSS3之3D变换实例详解
    移动端行列布局
    SVG描边动画原理
  • 原文地址:https://www.cnblogs.com/qingergege/p/5768376.html
Copyright © 2011-2022 走看看