springboot业务功能实战(十九)实现将文件或者文件夹压缩成zip

举报
小鲍侃java 发表于 2021/09/09 23:30:16 2021/09/09
【摘要】 最近碰到个需要下载zip压缩包的需求,于是我在网上找了下别人写好的zip工具类。但找了好多篇博客,总是发现有bug。因此就自己来写了个工具类。  这个工具类的功能为: (1)可以压缩文件,也可以压缩文件夹(5)代码中提供了2个压缩文件的方法,一个的输入参数为文件夹路径,一个为文件列表,可根据实际需求选择方法。(4)可以...

最近碰到个需要下载zip压缩包的需求,于是我在网上找了下别人写好的zip工具类。但找了好多篇博客,总是发现有bug。因此就自己来写了个工具类。

 这个工具类的功能为:

  • (1)可以压缩文件,也可以压缩文件夹
  • (5)代码中提供了2个压缩文件的方法,一个的输入参数为文件夹路径,一个为文件列表,可根据实际需求选择方法。
  • (4)可以选择是否保留原来的目录结构,如果不保留,所有文件跑压缩包根目录去了,且空文件夹直接舍弃。注意:如果不保留文件原来目录结构,在碰到文件名相同的文件时,会压缩失败。
  • (3)碰到空的文件夹,也可以压缩
  • (2)同时支持压缩多级文件夹,工具内部做了递归处理

 下面直接上代码

一、代码


  
  1. import java.io.*;
  2. import java.util.List;
  3. import java.util.zip.ZipEntry;
  4. import java.util.zip.ZipOutputStream;
  5. public class demo {
  6. private static final int BUFFER_SIZE = 2 * 1024;
  7. /**
  8. * 压缩成ZIP 方法1
  9. *
  10. * @param srcDir 压缩文件夹路径
  11. * @param out 压缩文件输出流
  12. * @param KeepDirStructure 是否保留原来的目录结构,true:保留目录结构;
  13. * false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)
  14. * @throws RuntimeException 压缩失败会抛出运行时异常
  15. */
  16. public static void toZip(String srcDir, OutputStream out, boolean KeepDirStructure) throws RuntimeException {
  17. long start = System.currentTimeMillis();
  18. ZipOutputStream zos = null;
  19. try {
  20. zos = new ZipOutputStream(out);
  21. File sourceFile = new File(srcDir);
  22. compress(sourceFile, zos, sourceFile.getName(), KeepDirStructure);
  23. long end = System.currentTimeMillis();
  24. System.out.println("压缩完成,耗时:" + (end - start) + " ms");
  25. } catch (Exception e) {
  26. throw new RuntimeException("zip error from ZipUtils", e);
  27. } finally {
  28. if (zos != null) {
  29. try {
  30. zos.close();
  31. } catch (IOException e) {
  32. e.printStackTrace();
  33. }
  34. }
  35. }
  36. }
  37. /**
  38. * 压缩成ZIP 方法2
  39. *
  40. * @param srcFiles 需要压缩的文件列表
  41. * @param out 压缩文件输出流
  42. * @throws RuntimeException 压缩失败会抛出运行时异常
  43. */
  44. public static void toZip(List<File> srcFiles, OutputStream out) throws RuntimeException {
  45. long start = System.currentTimeMillis();
  46. ZipOutputStream zos = null;
  47. try {
  48. zos = new ZipOutputStream(out);
  49. for (File srcFile : srcFiles) {
  50. byte[] buf = new byte[BUFFER_SIZE];
  51. zos.putNextEntry(new ZipEntry(srcFile.getName()));
  52. int len;
  53. FileInputStream in = new FileInputStream(srcFile);
  54. while ((len = in.read(buf)) != -1) {
  55. zos.write(buf, 0, len);
  56. }
  57. zos.closeEntry();
  58. in.close();
  59. }
  60. long end = System.currentTimeMillis();
  61. System.out.println("压缩完成,耗时:" + (end - start) + " ms");
  62. } catch (Exception e) {
  63. throw new RuntimeException("zip error from ZipUtils", e);
  64. } finally {
  65. if (zos != null) {
  66. try {
  67. zos.close();
  68. } catch (IOException e) {
  69. e.printStackTrace();
  70. }
  71. }
  72. }
  73. }
  74. /**
  75. * 递归压缩方法
  76. *
  77. * @param sourceFile 源文件
  78. * @param zos zip输出流
  79. * @param name 压缩后的名称
  80. * @param KeepDirStructure 是否保留原来的目录结构,true:保留目录结构;
  81. * false:所有文件跑到压缩包根目录下(注意:不保留目录结构可能会出现同名文件,会压缩失败)
  82. * @throws Exception
  83. */
  84. private static void compress(File sourceFile, ZipOutputStream zos, String name, boolean KeepDirStructure)
  85. throws Exception {
  86. byte[] buf = new byte[BUFFER_SIZE];
  87. if (sourceFile.isFile()) {
  88. // 向zip输出流中添加一个zip实体,构造器中name为zip实体的文件的名字
  89. zos.putNextEntry(new ZipEntry(name));
  90. // copy文件到zip输出流中
  91. int len;
  92. FileInputStream in = new FileInputStream(sourceFile);
  93. while ((len = in.read(buf)) != -1) {
  94. zos.write(buf, 0, len);
  95. }
  96. // Complete the entry
  97. zos.closeEntry();
  98. in.close();
  99. } else {
  100. File[] listFiles = sourceFile.listFiles();
  101. if (listFiles == null || listFiles.length == 0) {
  102. // 需要保留原来的文件结构时,需要对空文件夹进行处理
  103. if (KeepDirStructure) {
  104. // 空文件夹的处理
  105. zos.putNextEntry(new ZipEntry(name + "/"));
  106. // 没有文件,不需要文件的copy
  107. zos.closeEntry();
  108. }
  109. } else {
  110. for (File file : listFiles) {
  111. // 判断是否需要保留原来的文件结构
  112. if (KeepDirStructure) {
  113. // 注意:file.getName()前面需要带上父文件夹的名字加一斜杠,
  114. // 不然最后压缩包中就不能保留原来的文件结构,即:所有文件都跑到压缩包根目录下了
  115. compress(file, zos, name + "/" + file.getName(), KeepDirStructure);
  116. } else {
  117. compress(file, zos, file.getName(), KeepDirStructure);
  118. }
  119. }
  120. }
  121. }
  122. }
  123. public static void main(String[] args) throws Exception {
  124. /** 测试压缩方法1 */
  125. FileOutputStream fos1 = new FileOutputStream(new File("c:/mytest01.zip"));
  126. demo.toZip("D:\\common_files\\online", fos1, true);
  127. /** 测试压缩方法2 */
  128. // List<File> fileList = new ArrayList<>();
  129. // fileList.add(new File("D:/Java/jdk1.7.0_45_64bit/bin/jar.exe"));
  130. // fileList.add(new File("D:/Java/jdk1.7.0_45_64bit/bin/java.exe"));
  131. // FileOutputStream fos2 = new FileOutputStream(new File("c:/mytest02.zip"));
  132. //
  133. // demo.toZip(fileList, fos2);
  134. }
  135. }

二、注意事项

    写该工具类时,有些注意事项说一下:

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

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

    这个工具类在web项目中的使用场景就是多文件下载,我就简单说个下载多个excel表格的案例吧。

   代码中的步骤为:

  • (1)创建一个临时文件夹
  • (2)将要下载的文件生成至该临时文件夹内
  • (3)当所有文件生成完后,获取HttpServletResponse获取设置下载的header
  • (4)调用工具类的方法,传入上面生成的临时文件夹路径及response获取的输出流;这样就下载出来zip包了
  • (5)递归删除掉上面生成的临时文件夹和文件

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


  
  1. if(userList.size() > 0){
  2. /** 下面为下载zip压缩包相关流程 */
  3. HttpServletRequest request = ServletActionContext.getRequest();
  4. FileWriter writer;
  5. /** 1.创建临时文件夹 */
  6. String rootPath = request.getSession().getServletContext().getRealPath("/");
  7. File temDir = new File(rootPath + "/" + UUID.randomUUID().toString().replaceAll("-", ""));
  8. if(!temDir.exists()){
  9. temDir.mkdirs();
  10. }
  11. /** 2.生成需要下载的文件,存放在临时文件夹内 */
  12. // 这里我们直接来10个内容相同的文件为例,但这个10个文件名不可以相同
  13. for (int i = 0; i < 10; i++) {
  14. dataMap.put("userList", userList);
  15. Map<String, String> endMap = new HashMap<>();
  16. endMap.put("user", "老王");
  17. endMap.put("time", "2017-10-10 10:50:55");
  18. dataMap.put("endMap", endMap);
  19. Configuration cfg = new Configuration(Configuration.VERSION_2_3_22);
  20. cfg.setServletContextForTemplateLoading(ServletActionContext.getServletContext(), "/ftl");
  21. Template template = cfg.getTemplate("exportExcel.ftl");
  22. writer = new FileWriter(temDir.getPath()+"/excel"+ i +".xls");
  23. template.process(dataMap, writer);
  24. writer.flush();
  25. writer.close();
  26. }
  27. /** 3.设置response的header */
  28. HttpServletResponse response = ServletActionContext.getResponse();
  29. response.setContentType("application/zip");
  30. response.setHeader("Content-Disposition", "attachment; filename=excel.zip");
  31. /** 4.调用工具类,下载zip压缩包 */
  32. // 这里我们不需要保留目录结构
  33. demo.toZip(temDir.getPath(), response.getOutputStream(),false);
  34. /** 5.删除临时文件和文件夹 */
  35. // 这里我没写递归,直接就这样删除了
  36. File[] listFiles = temDir.listFiles();
  37. for (int i = 0; i < listFiles.length; i++) {
  38. listFiles[i].delete();
  39. }
  40. temDir.delete();
  41. }

文章来源: baocl.blog.csdn.net,作者:小黄鸡1992,版权归原作者所有,如需转载,请联系作者。

原文链接:baocl.blog.csdn.net/article/details/106765340

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。