Java实现将文件或者文件夹压缩成zip的详细代码

 更新时间:2021年11月01日 10:43:14   作者:zeng1994  
这篇文章主要介绍了Java实现将文件或者文件夹压缩成zip的详细代码,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下

最近碰到个需要下载zip压缩包的需求,于是我在网上找了下别人写好的zip工具类。但找了好多篇博客,总是发现有bug。因此就自己来写了个工具类。
        这个工具类的功能为:
(1)可以压缩文件,也可以压缩文件夹
(2)同时支持压缩多级文件夹,工具内部做了递归处理
(3)碰到空的文件夹,也可以压缩
(4)可以选择是否保留原来的目录结构,如果不保留,所有文件跑压缩包根目录去了,且空文件夹直接舍弃。注意:如果不保留文件原来目录结构,在碰到文件名相同的文件时,会压缩失败。
(5)代码中提供了2个压缩文件的方法,一个的输入参数为文件夹路径,一个为文件列表,可根据实际需求选择方法。

下面直接上代码

一、代码

package com.tax.core.util; 

import java.io.File;

import java.io.FileInputStream;

import java.io.FileOutputStream;

import java.io.IOException;

import java.io.OutputStream;

import java.util.ArrayList;

import java.util.List;

import java.util.zip.ZipEntry;

import java.util.zip.ZipOutputStream;

/**

 * ZipUtils

 * @author  ZENG.XIAO.YAN

 * @date    2017年11月19日 下午7:16:08

 * @version v1.0

 */

public class ZipUtils {

    

    private static final int  BUFFER_SIZE = 2 * 1024;

    /**

     * 压缩成ZIP 方法1

     * @param srcDir 压缩文件夹路径 

     * @param out    压缩文件输出流

     * @param KeepDirStructure  是否保留原来的目录结构,true:保留目录结构; 

     *                          false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)

     * @throws RuntimeException 压缩失败会抛出运行时异常

     */

    public static void toZip(String srcDir, OutputStream out, boolean KeepDirStructure)
           throws RuntimeException{

        

        long start = System.currentTimeMillis();

        ZipOutputStream zos = null ;

        try {

            zos = new ZipOutputStream(out);

            File sourceFile = new File(srcDir);

            compress(sourceFile,zos,sourceFile.getName(),KeepDirStructure);

            long end = System.currentTimeMillis();

            System.out.println("压缩完成,耗时:" + (end - start) +" ms");

        } catch (Exception e) {

            throw new RuntimeException("zip error from ZipUtils",e);

        }finally{

            if(zos != null){

                try {

                    zos.close();

                } catch (IOException e) {

                    e.printStackTrace();

                }

            }

        }

        

    }

    

    /**

     * 压缩成ZIP 方法2

     * @param srcFiles 需要压缩的文件列表

     * @param out           压缩文件输出流

     * @throws RuntimeException 压缩失败会抛出运行时异常

     */

    public static void toZip(List<File> srcFiles , OutputStream out)throws RuntimeException {

        long start = System.currentTimeMillis();

        ZipOutputStream zos = null ;

        try {

            zos = new ZipOutputStream(out);

            for (File srcFile : srcFiles) {

                byte[] buf = new byte[BUFFER_SIZE];

                zos.putNextEntry(new ZipEntry(srcFile.getName()));

                int len;

                FileInputStream in = new FileInputStream(srcFile);

                while ((len = in.read(buf)) != -1){

                    zos.write(buf, 0, len);

                }

                zos.closeEntry();

                in.close();

            }

            long end = System.currentTimeMillis();

            System.out.println("压缩完成,耗时:" + (end - start) +" ms");

        } catch (Exception e) {

            throw new RuntimeException("zip error from ZipUtils",e);

        }finally{

            if(zos != null){

                try {

                    zos.close();

                } catch (IOException e) {

                    e.printStackTrace();

                }

            }

        }

    }

    

    

    /**

     * 递归压缩方法

     * @param sourceFile 源文件

     * @param zos        zip输出流

     * @param name       压缩后的名称

     * @param KeepDirStructure  是否保留原来的目录结构,true:保留目录结构; 

     *                          false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)

     * @throws Exception

     */

    private static void compress(File sourceFile, ZipOutputStream zos, String name,

            boolean KeepDirStructure) throws Exception{

        byte[] buf = new byte[BUFFER_SIZE];

        if(sourceFile.isFile()){

            // 向zip输出流中添加一个zip实体,构造器中name为zip实体的文件的名字

            zos.putNextEntry(new ZipEntry(name));

            // copy文件到zip输出流中

            int len;

            FileInputStream in = new FileInputStream(sourceFile);

            while ((len = in.read(buf)) != -1){

                zos.write(buf, 0, len);

            }

            // Complete the entry

            zos.closeEntry();

            in.close();

        } else {

            File[] listFiles = sourceFile.listFiles();

            if(listFiles == null || listFiles.length == 0){

                // 需要保留原来的文件结构时,需要对空文件夹进行处理

                if(KeepDirStructure){

                    // 空文件夹的处理

                    zos.putNextEntry(new ZipEntry(name + "/"));

                    // 没有文件,不需要文件的copy

                    zos.closeEntry();

                }

                

            }else {

                for (File file : listFiles) {

                    // 判断是否需要保留原来的文件结构

                    if (KeepDirStructure) {

                        // 注意:file.getName()前面需要带上父文件夹的名字加一斜杠,

                        // 不然最后压缩包中就不能保留原来的文件结构,即:所有文件都跑到压缩包根目录下了

                        compress(file, zos, name + "/" + file.getName(),KeepDirStructure);

                    } else {

                        compress(file, zos, file.getName(),KeepDirStructure);

                    }

                    

                }

            }

        }

    }

    public static void main(String[] args) throws Exception {

        /** 测试压缩方法1  */

        FileOutputStream fos1 = new FileOutputStream(new File("c:/mytest01.zip"));

        ZipUtils.toZip("D:/log", fos1,true);

        /** 测试压缩方法2  */

        List<File> fileList = new ArrayList<>();

        fileList.add(new File("D:/Java/jdk1.7.0_45_64bit/bin/jar.exe"));

        fileList.add(new File("D:/Java/jdk1.7.0_45_64bit/bin/java.exe"));

        FileOutputStream fos2 = new FileOutputStream(new File("c:/mytest02.zip"));

        ZipUtils.toZip(fileList, fos2);

    }

}

二、注意事项

写该工具类时,有些注意事项说一下:
        (1)支持选择是否保留原来的文件目录结构,如果不保留,那么空文件夹直接不用处理。
        (1)碰到空文件夹时,如果需要保留目录结构,则直接添加个ZipEntry就可以了,不过就是这个entry的名字后面需要带上一斜杠(/)表示这个是目录。
        (2)递归时,不需要把zip输出流关闭,zip输出流的关闭应该是在调用完递归方法后面关闭
        (3)递归时,如果是个文件夹且需要保留目录结构,那么在调用方法压缩他的子文件时,需要把文件夹的名字加一斜杠给添加到子文件名字前面,这样压缩后才有多级目录。

三、如何在javaWeb项目中使用该工具类

    这个工具类在web项目中的使用场景就是多文件下载,我就简单说个下载多个excel表格的案例吧。
    代码中的步骤为:
        (1)创建一个临时文件夹
        (2)将要下载的文件生成至该临时文件夹内
        (3)当所有文件生成完后,获取HttpServletResponse获取设置下载的header
        (4)调用工具类的方法,传入上面生成的临时文件夹路径及response获取的输出流;这样就下载出来zip包了
        (5)递归删除掉上面生成的临时文件夹和文件

    下面为一个示例代码的代码片段,不是完整代码,简单看一下代码中的步骤

if(userList.size() > 0){

        /** 下面为下载zip压缩包相关流程 */

        HttpServletRequest request = ServletActionContext.getRequest();

        FileWriter writer;

        /** 1.创建临时文件夹  */

        String rootPath = request.getSession().getServletContext().getRealPath("/");

        File temDir = new File(rootPath + "/" + UUID.randomUUID().toString().replaceAll("-", ""));

        if(!temDir.exists()){

            temDir.mkdirs();

        }

        /** 2.生成需要下载的文件,存放在临时文件夹内 */

        // 这里我们直接来10个内容相同的文件为例,但这个10个文件名不可以相同

        for (int i = 0; i < 10; i++) {

            dataMap.put("userList", userList);

            Map<String, String> endMap = new HashMap<>();

            endMap.put("user", "老王");

            endMap.put("time", "2017-10-10 10:50:55");

            dataMap.put("endMap", endMap);

            Configuration cfg = new Configuration(Configuration.VERSION_2_3_22);

            cfg.setServletContextForTemplateLoading(ServletActionContext.getServletContext(), "/ftl");

            Template template = cfg.getTemplate("exportExcel.ftl");

            writer = new FileWriter(temDir.getPath()+"/excel"+ i +".xls");

            template.process(dataMap, writer);

            writer.flush();

            writer.close();

        }

        

        /** 3.设置response的header */

        HttpServletResponse response = ServletActionContext.getResponse();

        response.setContentType("application/zip");

        response.setHeader("Content-Disposition", "attachment; filename=excel.zip");  

        /** 4.调用工具类,下载zip压缩包 */

        // 这里我们不需要保留目录结构

        ZipUtils.toZip(temDir.getPath(), response.getOutputStream(),false);

   /** 5.删除临时文件和文件夹 */

        // 这里我没写递归,直接就这样删除了

        File[] listFiles = temDir.listFiles();

        for (int i = 0; i < listFiles.length; i++) {

            listFiles[i].delete();

        }

        temDir.delete();

    }

到此这篇关于Java实现将文件或者文件夹压缩成zip的文章就介绍到这了,更多相关Java实现将文件或者文件夹压缩成zip内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Spring Boot整合Mybatis并完成CRUD操作的实现示例

    Spring Boot整合Mybatis并完成CRUD操作的实现示例

    这篇文章主要介绍了Spring Boot整合Mybatis并完成CRUD操作的实现示例,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • mybatis插件pageHelper实现分页效果

    mybatis插件pageHelper实现分页效果

    这篇文章主要为大家详细介绍了mybatis插件pageHelper实现分页效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2018-12-12
  • Java数据结构之顺序表篇

    Java数据结构之顺序表篇

    顺序表,全名顺序存储结构,是线性表的一种。线性表用于存储逻辑关系为“一对一”的数据,顺序表自然也不例外,不仅如此,顺序表对数据物理存储结构也有要求。顺序表存储数据时,会提前申请一整块足够大小的物理空间,然后将数据依次存储起来,存储时数据元素间不留缝隙
    2022-01-01
  • Java关键字synchronized原理与锁的状态详解

    Java关键字synchronized原理与锁的状态详解

    在Java当中synchronized关键字通常是用来标记一个方法或者代码块。本文将通过示例为大家详细介绍一下Synchronized的各种使用方法,需要的可以参考一下
    2022-08-08
  • SpringBoot+ENC实现密钥加密的使用示例

    SpringBoot+ENC实现密钥加密的使用示例

    本文主要介绍了SpringBoot+ENC实现密钥加密的使用示例,主要是为了将配置信息从应用程序代码中分离出来,以提高安全性和可维护性,感兴趣的可以了解一下
    2024-07-07
  • idea中同一SpringBoot项目多端口启动

    idea中同一SpringBoot项目多端口启动

    本文主要介绍了idea中同一SpringBoot项目多端口启动,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • java中的HashMap多层嵌套

    java中的HashMap多层嵌套

    这篇文章主要介绍了java中的HashMap多层嵌套问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-09-09
  • 95%的Java程序员人都用不好Synchronized详解

    95%的Java程序员人都用不好Synchronized详解

    这篇文章主要为大家介绍了95%的Java程序员人都用不好Synchronized详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-03-03
  • Java接口幂等性设计原理解析

    Java接口幂等性设计原理解析

    这篇文章主要介绍了Java接口幂等性设计原理解析,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-05-05
  • MyBatis如何进行双重foreach循环

    MyBatis如何进行双重foreach循环

    这篇文章主要介绍了MyBatis如何进行双重foreach循环,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-02-02

最新评论