java对zip、rar、7z文件带密码解压实例

简介: 本文采用java语言实现了对zip和rar、7z文件的解压统一算法。并对比了相应的解压速度,支持传入密码进行在线解压。

      在一些日常业务中,会遇到一些琐碎文件需要统一打包到一个压缩包中上传,业务方在后台接收到压缩包后自行解压,然后解析相应文件。而且可能涉及安全保密,因此会在压缩时带上密码,要求后台业务可以指定密码进行解压。

     应用环境说明:jdk1.8,maven3.x,需要基于java语言实现对zip、rar、7z等常见压缩包的解压工作。

     首先关于zip和rar、7z等压缩工具和压缩算法就不在此赘述,下面通过一个数据对比,使用上述三种不同的压缩算法,采用默认的压缩方式,看到压缩的文件大小如下:

image.png

 转换成图表看得更直观,如下图:

image.png

     从以上图表可以看到,7z的压缩率是最高,而zip压缩率比较低,rar比zip稍微好点。单纯从压缩率看,7z>rar4>rar5>zip。

    下面具体说明在java中如何进行相应解压:

1、pom.xml

<project xmlns="http://maven.apache.org/POM/4.0.0"

xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"

xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">

<modelVersion>4.0.0</modelVersion>

<groupId>com.yelang</groupId>

<artifactId>7zdemo</artifactId>

<version>0.0.1-SNAPSHOT</version>

<dependencies>

 <dependency>

  <groupId>net.lingala.zip4j</groupId>

  <artifactId>zip4j</artifactId>

  <version>2.9.0</version>

 </dependency>

 <dependency>

  <groupId>net.sf.sevenzipjbinding</groupId>

  <artifactId>sevenzipjbinding</artifactId>

  <version>16.02-2.01</version>

 </dependency>

 <dependency>

  <groupId>net.sf.sevenzipjbinding</groupId>

  <artifactId>sevenzipjbinding-all-platforms</artifactId>

  <version>16.02-2.01</version>

 </dependency>

 <dependency>

  <groupId>org.tukaani</groupId>

  <artifactId>xz</artifactId>

  <version>1.9</version>

 </dependency>

 <dependency>

  <groupId>org.apache.commons</groupId>

  <artifactId>commons-compress</artifactId>

  <version>1.21</version>

 </dependency>

 

 <!-- https://mvnrepository.com/artifact/org.slf4j/slf4j-api -->

 <dependency>

     <groupId>org.slf4j</groupId>

     <artifactId>slf4j-api</artifactId>

     <version>1.7.30</version>

 </dependency>

 

 <!-- https://mvnrepository.com/artifact/org.apache.commons/commons-lang3 -->

 <dependency>

     <groupId>org.apache.commons</groupId>

     <artifactId>commons-lang3</artifactId>

     <version>3.12.0</version>

 </dependency>

 

 <!-- https://mvnrepository.com/artifact/fr.opensagres.xdocreport/xdocreport -->

 <dependency>

     <groupId>fr.opensagres.xdocreport</groupId>

     <artifactId>xdocreport</artifactId>

     <version>1.0.6</version>

 </dependency>

 

</dependencies>

</project>

主要依赖的jar包有:zip4j、sevenzipjbinding等。

2、zip解压


@SuppressWarnings("resource")

private static String unZip(String rootPath, String sourceRarPath, String destDirPath, String passWord) {

       ZipFile zipFile = null;

       String result = "";

       try {

        //String filePath = sourceRarPath;

        String filePath = rootPath + sourceRarPath;

           if (StringUtils.isNotBlank(passWord)) {

               zipFile = new ZipFile(filePath, passWord.toCharArray());

           } else {

               zipFile = new ZipFile(filePath);

           }

           zipFile.setCharset(Charset.forName("GBK"));

           zipFile.extractAll(rootPath + destDirPath);

       } catch (Exception e) {

           log.error("unZip error", e);

           return e.getMessage();

       }

       return result;

   }


3、rar解压


private static String unRar(String rootPath, String sourceRarPath, String destDirPath, String passWord) {

       String rarDir = rootPath + sourceRarPath;

       String outDir = rootPath + destDirPath + File.separator;

       RandomAccessFile randomAccessFile = null;

       IInArchive inArchive = null;

       try {

           // 第一个参数是需要解压的压缩包路径,第二个参数参考JdkAPI文档的RandomAccessFile

           randomAccessFile = new RandomAccessFile(rarDir, "r");

           if (StringUtils.isNotBlank(passWord))

               inArchive = SevenZip.openInArchive(null, new RandomAccessFileInStream(randomAccessFile), passWord);

           else

               inArchive = SevenZip.openInArchive(null, new RandomAccessFileInStream(randomAccessFile));

           ISimpleInArchive simpleInArchive = inArchive.getSimpleInterface();

           for (final ISimpleInArchiveItem item : simpleInArchive.getArchiveItems()) {

               final int[] hash = new int[]{0};

               if (!item.isFolder()) {

                   ExtractOperationResult result;

                   final long[] sizeArray = new long[1];

                   File outFile = new File(outDir + item.getPath());

                   File parent = outFile.getParentFile();

                   if ((!parent.exists()) && (!parent.mkdirs())) {

                       continue;

                   }

                   if (StringUtils.isNotBlank(passWord)) {

                       result = item.extractSlow(data -> {

                           try {

                               IOUtils.write(data, new FileOutputStream(outFile, true));

                           } catch (Exception e) {

                               e.printStackTrace();

                           }

                           hash[0] ^= Arrays.hashCode(data); // Consume data

                           sizeArray[0] += data.length;

                           return data.length; // Return amount of consumed

                       }, passWord);

                   } else {

                       result = item.extractSlow(data -> {

                           try {

                               IOUtils.write(data, new FileOutputStream(outFile, true));

                           } catch (Exception e) {

                               e.printStackTrace();

                           }

                           hash[0] ^= Arrays.hashCode(data); // Consume data

                           sizeArray[0] += data.length;

                           return data.length; // Return amount of consumed

                       });

                   }

                 

                   if (result == ExtractOperationResult.OK) {

                       log.error("解压rar成功...." + String.format("%9X | %10s | %s", hash[0], sizeArray[0], item.getPath()));

                   } else if (StringUtils.isNotBlank(passWord)) {

                       log.error("解压rar成功:密码错误或者其他错误...." + result);

                       return "password";

                   } else {

                       return "rar error";

                   }

               }

           }

       } catch (Exception e) {

           log.error("unRar error", e);

           return e.getMessage();

       } finally {

           try {

               inArchive.close();

               randomAccessFile.close();

           } catch (Exception e) {

               e.printStackTrace();

           }

       }

       return "";

   }


4、7z解压


private static String un7z(String rootPath, String sourceRarPath, String destDirPath, String passWord) {

       try {

           File srcFile = new File(rootPath + sourceRarPath);//获取当前压缩文件

           // 判断源文件是否存在

           if (!srcFile.exists()) {

               throw new Exception(srcFile.getPath() + "所指文件不存在");

           }

           //开始解压

           SevenZFile zIn = null;

           if (StringUtils.isNotBlank(passWord)) {

            zIn = new SevenZFile(srcFile, passWord.toCharArray());

           }  else {

            zIn = new SevenZFile(srcFile);

           }

           SevenZArchiveEntry entry = null;

           File file = null;

           while ((entry = zIn.getNextEntry()) != null) {

               if (!entry.isDirectory()) {

                   file = new File(rootPath + destDirPath, entry.getName());

                   if (!file.exists()) {

                       new File(file.getParent()).mkdirs();//创建此文件的上级目录

                   }

                   OutputStream out = new FileOutputStream(file);

                   BufferedOutputStream bos = new BufferedOutputStream(out);

                   int len = -1;

                   byte[] buf = new byte[1024];

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

                       bos.write(buf, 0, len);

                   }

                   // 关流顺序,先打开的后关闭

                   bos.close();

                   out.close();

               }

           }

       } catch (Exception e) {

           log.error("un7z is error", e);

           return e.getMessage();

       }

       return "";

   }


5、解压统一入口封装


public static Map<String,Object> unFile(String rootPath, String sourcePath, String destDirPath, String passWord) {

    Map<String,Object> resultMap = new HashMap<String, Object>();

    String result = "";

       if (sourcePath.toLowerCase().endsWith(".zip")) {

           //Wrong password!

           result = unZip(rootPath, sourcePath, destDirPath, passWord);

       } else if (sourcePath.toLowerCase().endsWith(".rar")) {

           //java.security.InvalidAlgorithmParameterException: password should be specified

           result = unRar(rootPath, sourcePath, destDirPath, passWord);

           System.out.println(result);

       } else if (sourcePath.toLowerCase().endsWith(".7z")) {

           //PasswordRequiredException: Cannot read encrypted content from G:\ziptest\11111111.7z without a password

           result = un7z(rootPath, sourcePath, destDirPath, passWord);

       }

 

       resultMap.put("resultMsg", 1);

       if (StringUtils.isNotBlank(result)) {

           if (result.contains("password")) resultMap.put("resultMsg", 2);

           if (!result.contains("password")) resultMap.put("resultMsg", 3);

       }

       resultMap.put("files", null);

       //System.out.println(result + "==============");

       return resultMap;

   }


6、测试代码


Long start = System.currentTimeMillis();

unFile("D:/rarfetch0628/","apache-tomcat-8.5.69.zip","apache-tomcat-zip","222");

long end = System.currentTimeMillis();

System.out.println("zip解压耗时==" + (end - start) + "毫秒");

System.out.println("============================================================");

 

Long rar4start = System.currentTimeMillis();

unFile("D:/rarfetch0628/","apache-tomcat-8.5.69-4.rar","apache-tomcat-rar4","222");

long rar4end = System.currentTimeMillis();

System.out.println("rar4解压耗时==" + (rar4end - rar4start)+ "毫秒");

System.out.println("============================================================");

 

Long rar5start = System.currentTimeMillis();

unFile("D:/rarfetch0628/","apache-tomcat-8.5.69-5.rar","apache-tomcat-rar5","222");

long rar5end = System.currentTimeMillis();

System.out.println("rar5解压耗时==" + (rar5end - rar5start)+ "毫秒");

System.out.println("============================================================");

 

Long zstart = System.currentTimeMillis();

unFile("D:/rarfetch0628/","apache-tomcat-8.5.69.7z","apache-tomcat-7z","222");

long zend = System.currentTimeMillis();

System.out.println("7z解压耗时==" + (zend - zstart)+ "毫秒");

System.out.println("============================================================");


在控制台中可以看到以下结果:

image.png

image.png

     总结:本文采用java语言实现了对zip和rar、7z文件的解压统一算法。并对比了相应的解压速度,支持传入密码进行在线解压。本文参考了:https://blog.csdn.net/luanwuqingyang/article/details/121114113,不过博主的文章代码直接运行有问题,这里进行了调整,主要优化的点如下:

1、pom.xml 遗漏了slf4j、commons-lang3、xdocreport等依赖

2、zip路径优化

3、去掉一些无用信息

4、优化异常信息

目录
相关文章
|
2月前
|
Java
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
java小工具util系列5:java文件相关操作工具,包括读取服务器路径下文件,删除文件及子文件,删除文件夹等方法
77 9
|
2月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
86 2
|
1天前
|
人工智能 自然语言处理 Java
FastExcel:开源的 JAVA 解析 Excel 工具,集成 AI 通过自然语言处理 Excel 文件,完全兼容 EasyExcel
FastExcel 是一款基于 Java 的高性能 Excel 处理工具,专注于优化大规模数据处理,提供简洁易用的 API 和流式操作能力,支持从 EasyExcel 无缝迁移。
33 9
FastExcel:开源的 JAVA 解析 Excel 工具,集成 AI 通过自然语言处理 Excel 文件,完全兼容 EasyExcel
|
22天前
|
Java
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
java实现从HDFS上下载文件及文件夹的功能,以流形式输出,便于用户自定义保存任何路径下
82 34
|
2月前
|
消息中间件 存储 Java
RocketMQ文件刷盘机制深度解析与Java模拟实现
【11月更文挑战第22天】在现代分布式系统中,消息队列(Message Queue, MQ)作为一种重要的中间件,扮演着连接不同服务、实现异步通信和消息解耦的关键角色。Apache RocketMQ作为一款高性能的分布式消息中间件,广泛应用于实时数据流处理、日志流处理等场景。为了保证消息的可靠性,RocketMQ引入了一种称为“刷盘”的机制,将消息从内存写入到磁盘中,确保消息持久化。本文将从底层原理、业务场景、概念、功能点等方面深入解析RocketMQ的文件刷盘机制,并使用Java模拟实现类似的功能。
43 3
|
2月前
|
Java 测试技术 Maven
Maven clean 提示文件 java.io.IOException
在使用Maven进行项目打包时,遇到了`Failed to delete`错误,尝试手动删除目标文件也失败,提示`java.io.IOException`。经过分析,发现问题是由于`sys-info.log`文件被其他进程占用。解决方法是关闭IDEA和相关Java进程,清理隐藏的Java进程后重新尝试Maven clean操作。最终问题得以解决。总结:遇到此类问题时,可以通过任务管理器清理相关进程或重启电脑来解决。
|
2月前
|
存储 缓存 安全
在 Java 编程中,创建临时文件用于存储临时数据或进行临时操作非常常见
在 Java 编程中,创建临时文件用于存储临时数据或进行临时操作非常常见。本文介绍了使用 `File.createTempFile` 方法和自定义创建临时文件的两种方式,详细探讨了它们的使用场景和注意事项,包括数据缓存、文件上传下载和日志记录等。强调了清理临时文件、确保文件名唯一性和合理设置文件权限的重要性。
112 2
|
6天前
|
Java
Java—多线程实现生产消费者
本文介绍了多线程实现生产消费者模式的三个版本。Version1包含四个类:`Producer`(生产者)、`Consumer`(消费者)、`Resource`(公共资源)和`TestMain`(测试类)。通过`synchronized`和`wait/notify`机制控制线程同步,但存在多个生产者或消费者时可能出现多次生产和消费的问题。 Version2将`if`改为`while`,解决了多次生产和消费的问题,但仍可能因`notify()`随机唤醒线程而导致死锁。因此,引入了`notifyAll()`来唤醒所有等待线程,但这会带来性能问题。
Java—多线程实现生产消费者
|
8天前
|
安全 Java Kotlin
Java多线程——synchronized、volatile 保障可见性
Java多线程中,`synchronized` 和 `volatile` 关键字用于保障可见性。`synchronized` 保证原子性、可见性和有序性,通过锁机制确保线程安全;`volatile` 仅保证可见性和有序性,不保证原子性。代码示例展示了如何使用 `synchronized` 和 `volatile` 解决主线程无法感知子线程修改共享变量的问题。总结:`volatile` 确保不同线程对共享变量操作的可见性,使一个线程修改后,其他线程能立即看到最新值。
|
8天前
|
消息中间件 缓存 安全
Java多线程是什么
Java多线程简介:本文介绍了Java中常见的线程池类型,包括`newCachedThreadPool`(适用于短期异步任务)、`newFixedThreadPool`(适用于固定数量的长期任务)、`newScheduledThreadPool`(支持定时和周期性任务)以及`newSingleThreadExecutor`(保证任务顺序执行)。同时,文章还讲解了Java中的锁机制,如`synchronized`关键字、CAS操作及其实现方式,并详细描述了可重入锁`ReentrantLock`和读写锁`ReadWriteLock`的工作原理与应用场景。