Java 压缩多个文件为zip包(中间不生成临时文件,直接压缩为zip二进制流),以及解压zip包二进制流为文件

Java 压缩多个文件为zip包及解压zip包以及压缩多文件为zip文件流解压zip二进制流(中间不生成临时文件,直接压缩为zip二进制流,并验证解压)

    • 1. 效果图
    • 2. 源码

这篇博客将提供俩种方法,

  1. 提前生成要压缩的多个文件,然后读取文件夹多层或一层去遍历压缩zip包
  2. 直接用原始文件名称及二进制流,压缩返回zip包二进制流,中间不生成冗余文件;
    很明显方法2更优一些;
  3. 解压zip文件或者zip文件流验证;

1. 效果图

压缩俩个文件到zip包,并分别解析zip包文件及zip二进制流,打印文件及文件内容详情效果图如下:

在这里插入图片描述

2. 源码

import java.io.*;
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
import java.util.zip.ZipOutputStream;public class FileUtil {public static void main(String[] args) throws Exception {long timeStamp = System.currentTimeMillis();String uuid = UUID.randomUUID().toString();// 生成本地文件,然后压缩,压缩完成清除本地文件及冗余文件夹zipTest1(uuid, timeStamp);// 不生成中间文件,直接使用文件名称+文件流进行压缩 更优zipTest2(uuid, timeStamp);}private static void zipTest2(String uuid, long timeStamp) throws IOException {String zipFileName = String.format("%s_%s.zip", uuid, timeStamp);String imgFileName = String.format("%s_%s_%s.json", "img", uuid, timeStamp);String referInfoFile = String.format("%s_%s_%s.json", "ref", uuid, timeStamp);String[] fileNames = new String[]{imgFileName, referInfoFile};List<byte[]> bytesList = new ArrayList<>();byte[] file0 = imgFileName.toString().getBytes();byte[] file1 = referInfoFile.toString().getBytes();bytesList.add(file0);bytesList.add(file1);// zip压缩流byte[] rodZip = packageZipToBytes(fileNames, bytesList);readAndParseZip(zipFileName, rodZip);}private static void zipTest1(String uuid, long timeStamp) throws Exception {String path = System.getProperty("user.dir") + File.separator + uuid + "_" + timeStamp + File.separator;String resPath = System.getProperty("user.dir") + File.separator + "out" + File.separator + uuid + "_" + timeStamp;deleteDirectory(path);delOrCreateDir(path);delOrCreateDir(resPath);String imgFileName = String.format("%s%s_%s_%s.json", path, "img", uuid, timeStamp);String referInfoFile = String.format("%s%s_%s_%s.json", path, "ref", uuid, timeStamp);try (FileWriter fileWriter = new FileWriter(referInfoFile)) {fileWriter.write(referInfoFile.toString());}try (FileWriter fileWriter = new FileWriter(imgFileName)) {fileWriter.write(imgFileName.toString());}System.out.println("zip start ---------------" + System.currentTimeMillis());FileUtil test3 = new FileUtil();String packagePath = path;  //选中的文件夹test3.packageZip(packagePath, resPath + ".zip");System.out.println("zip finish ---------------" + System.currentTimeMillis());// 删除冗余文件,文件夹deleteDirectory(path);deleteDirectory(resPath);deleteDirectory(resPath);}/*** 压缩多个文件为zip包文件流** @param fileNames 文件名称* @param byteLists 文件流* @return*/public static byte[] packageZipToBytes(String[] fileNames, List<byte[]> byteLists) throws IOException {ByteArrayOutputStream bos = new ByteArrayOutputStream();ZipOutputStream zipOutputStream = null;try {zipOutputStream = new ZipOutputStream(bos);for (int i = 0; i < fileNames.length; i++) {String fileName = fileNames[i];fileName = fileName.substring(fileName.lastIndexOf(File.separator) + 1);byte[] content = byteLists.get(i);// 依次对每个文件都压缩try {zipOutputStream.putNextEntry(new ZipEntry(fileName));zipOutputStream.write(content);} catch (IOException e) {throw new RuntimeException(e);}}zipOutputStream.close();bos.close();} catch (FileNotFoundException e) {throw new RuntimeException(e);} catch (IOException e) {throw new RuntimeException(e);}byte[] zipOutBytes = bos.toByteArray();
/*        OutputStream fos = new FileOutputStream("D:\\study\\ark-common-model\\out\\aaaaaaaa_1689583754986.zip");fos.write(zipOutBytes, 0, zipOutBytes.length);fos.close();*/return zipOutBytes;}private static void readAndParseZip(String zipFilePath, byte[] rodZip) {ByteArrayInputStream fis = null;FileInputStream fileInputStream = null;ZipInputStream zis = null;try {if (rodZip != null) {fis = new ByteArrayInputStream(rodZip);zis = new ZipInputStream(fis);} else {fileInputStream = new FileInputStream(zipFilePath);zis = new ZipInputStream(fileInputStream);}ZipEntry zipEntry = zis.getNextEntry();while (zipEntry != null) {// 如果该项是一个文件if (!zipEntry.isDirectory()) {String fileName = zipEntry.getName();ByteArrayOutputStream bos = new ByteArrayOutputStream();byte[] buffer = new byte[1024];int len;while ((len = zis.read(buffer)) > 0) {bos.write(buffer, 0, len);}// 将解压出的文件流输出到控制台String content = bos.toString();System.out.println("fileName: " + fileName + ",content: " + content);}zis.closeEntry();zipEntry = zis.getNextEntry();}zis.close();fis.close();} catch (IOException e) {e.printStackTrace();}}public static void delOrCreateDir(String imgPath) {File file = new File(imgPath);if (!file.getParentFile().exists()) {//上级目录不存在,创建上级目录file.getParentFile().mkdirs();} else {
//            FileUtils.deleteFolder(imgPath);}file.mkdirs();}/*** 根据路径删除指定的目录或文件,无论存在与否** @param sPath 要删除的目录或文件* @return 删除成功返回 true,否则返回 false。*/public static boolean deleteFolder(String sPath) {boolean flag = false;File file = new File(sPath);// 判断目录或文件是否存在if (!file.exists()) {  // 不存在返回 falsereturn flag;} else {// 判断是否为文件if (file.isFile()) {  // 为文件时调用删除文件方法return deleteFile(sPath);} else {  // 为目录时调用删除目录方法return deleteDirectory(sPath);}}}/*** 删除单个文件** @param sPath 被删除文件的文件名* @return 单个文件删除成功返回true,否则返回false*/public static boolean deleteFile(String sPath) {boolean flag = false;File file = new File(sPath);// 路径为文件且不为空则进行删除if (file.isFile() && file.exists()) {file.delete();flag = true;}return flag;}/*** 删除目录(文件夹)以及目录下的文件** @param sPath 被删除目录的文件路径* @return 目录删除成功返回true,否则返回false*/public static boolean deleteDirectory(String sPath) {boolean flag = false;//如果sPath不以文件分隔符结尾,自动添加文件分隔符if (!sPath.endsWith(File.separator)) {sPath = sPath + File.separator;}File dirFile = new File(sPath);//如果dir对应的文件不存在,或者不是一个目录,则退出if (!dirFile.exists() || !dirFile.isDirectory()) {return false;}flag = true;//删除文件夹下的所有文件(包括子目录)File[] files = dirFile.listFiles();for (int i = 0; i < files.length; i++) {//删除子文件if (files[i].isFile()) {flag = deleteFile(files[i].getAbsolutePath());if (!flag) {break;}} //删除子目录else {flag = deleteDirectory(files[i].getAbsolutePath());if (!flag) {break;}}}if (!flag) {return false;}//删除当前目录if (dirFile.delete()) {
//            dirFile.getParentFile().delete();return true;} else {
//            dirFile.getParentFile().delete();return false;}}public void packageZip(String filesPath, String resPath) throws Exception {// 要被压缩的文件夹File file = new File(filesPath);   //需要压缩的文件夹File zipFile = new File(resPath);ZipOutputStream zipOut = new ZipOutputStream(new FileOutputStream(zipFile));isDirectory(file, zipOut, "", true);   //判断是否为文件夹zipOut.close();}public void isDirectory(File file, ZipOutputStream zipOutputStream, String filePath, boolean flag) throwsIOException {//判断是否为问加减if (file.isDirectory()) {File[] files = file.listFiles();  //获取该文件夹下所有文件(包含文件夹)filePath = flag == true ? file.getName() : filePath + File.separator + file.getName();   //首次为选中的文件夹,即根目录,之后递归实现拼接目录for (int i = 0; i < files.length; ++i) {//判断子文件是否为文件夹if (files[i].isDirectory()) {System.out.println("-----" + files[i].getName());//进入递归,flag置false 即当前文件夹下仍包含文件夹isDirectory(files[i], zipOutputStream, filePath, false);} else {System.out.println("fileName: " + files[i].getName());//不为文件夹则进行压缩InputStream input = new FileInputStream(files[i]);zipOutputStream.putNextEntry(new ZipEntry(files[i].getName()));int temp = 0;while ((temp = input.read()) != -1) {zipOutputStream.write(temp);}input.close();}}} else {//将子文件夹下的文件进行压缩InputStream input = new FileInputStream(file);zipOutputStream.putNextEntry(new ZipEntry(file.getPath()));int temp = 0;while ((temp = input.read()) != -1) {zipOutputStream.write(temp);}input.close();}}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/27845.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

招聘小程序制作:连接人才与企业

随着人才市场的竞争日益激烈&#xff0c;招聘小程序成为了企业寻找优秀人才和求职者找到理想工作的重要工具。通过招聘小程序&#xff0c;企业可以发布招聘信息、筛选简历&#xff0c;而求职者可以浏览职位、提交简历等。 招聘小程序的好处 精准匹配人才&#xff1a;招聘小程序…

js的this绑定规则以及箭头函数

目录 调用位置默认绑定隐式绑定隐式丢失 显式绑定callapplybind new绑定装箱绑定优先级this规则之外忽略显式绑定间接函数引用 箭头函数 调用位置 从字面意思上来理解&#xff0c;this似乎是指向自己的 然而在JavaScript中&#xff0c;this并不是绑定到自身的 可以看这一个例子…

经典常谈思维导图怎么制作?手把手教你制作

经典常谈思维导图怎么制作&#xff1f;创建思维导图可以帮助我们更好地组织和整理信息&#xff0c;帮助我们更好地理解和记忆信息。它可以使我们更高效地学习和工作&#xff0c;并帮助我们更好地表达和分享我们的想法和想法。因此&#xff0c;制作思维导图是一种非常有用的技能…

适合投资者的交易策略,4步找到

在外汇交易市场中&#xff0c;根据市场情况和个人投资目标&#xff0c;制定灵活的交易策略是至关重要的。Forexclub认为投资者可以通过结合多种交易策略&#xff0c;打造出全面、科学且适合自己的外汇交易策略。 首先&#xff0c;基于技术指标的交易策略是判断市场趋势和转折点…

剑指 Offer 29. 顺时针打印矩阵

输入一个矩阵&#xff0c;按照从外向里以顺时针的顺序依次打印出每一个数字。 示例 1&#xff1a; 输入&#xff1a;matrix [[1,2,3],[4,5,6],[7,8,9]] 输出&#xff1a;[1,2,3,6,9,8,7,4,5] 示例 2&#xff1a; 输入&#xff1a;matrix [[1,2,3,4],[5,6,7,8],[9,10,11,12]] …

【深度学习】张量的广播专题

一、说明 张量广播&#xff08;tensor broadcasting&#xff09;是一种将低维张量自动转化为高维张量的技术&#xff0c;使得张量之间可以进行基于元素的运算&#xff08;如加、减、乘等&#xff09;。在进行张量广播时&#xff0c;会将维度数较少的张量沿着长度为1的轴进行复制…

【深度学习】了解残差网 ResNet 和 ResNeXt 的架构

一、说明 了解和实现 ResNet 和 ResNeXt 的架构以实现最先进的图像分类&#xff1a;从Microsoft到 Facebook [第 1 部分]&#xff0c;在这篇由两部分组成的博客文章中&#xff0c;我们将探讨残差网络。更具体地说&#xff0c;我们将讨论Microsoft研究和Facebook AI研究发布的三…

【C++初阶】list的模拟实现 附源码

一.list介绍 list底层是一个双向带头循环链表&#xff0c;这个我们以前用C语言模拟实现过&#xff0c;->双向带头循环链表 下面是list的文档介绍&#xff1a; list文档介绍 我们会根据 list 的文档来模拟实现 list 的增删查改及其它接口。 二.list模拟实现思路 既然是用C模拟…

【Vue】给 elementUI 中的 this.$confirm、this.$alert、 this.$prompt添加按钮的加载效果

文章目录 主要使用 beforeClose 方法实现 loading 的效果beforeClose MessageBox 关闭前的回调&#xff0c;会暂停实例的关闭 function(action, instance, done)1. action 的值为confirm, cancel或close。 2. instance 为 MessageBox 实例&#xff0c;可以通过它访问实例上的属…

Django + Bootstrap - 【echart】 统计图表进阶使用-统计用户日活日增、月活月增等数据(二)

一. 前言 Bootstrap是一个流行的前端框架&#xff0c;而ECharts是一个流行的可视化库。 Bootstrap可以用来设计网站和应用程序的用户界面&#xff0c;而ECharts可以用来创建交互式和可视化的图表。 chart.js中文文档&#xff1a;http://www.bootcss.com/p/chart.js/docs/ 二. …

LT8619B 是一款HDMI转TTL或者2 PORT LVDS的芯片。

LT8619B 1. 概述 LT8619B是龙迅基于清除边缘技术的高性能HDMI接收芯片&#xff0c;符合HDMI 1.4&#xff08;高清多媒体接口&#xff09;规范。RGB 输出端口可支持 RGB888/RGB666/RGB565 格式&#xff0c;输出分辨率最高可支持 4Kx2K 分辨率。凭借可编程标量&#xff0c;LT86…

切换.net Framework 版本后,出现NuGet 包是使用不同于当前目标框架的目标框架安装的,可能需要重新安装

问题现象&#xff1a; 由于添加新的dll文件&#xff0c;依赖的.NET Framework版本与当前的不一致&#xff0c;在vs 中切换了目标框架版本后&#xff0c;运行程序&#xff0c;出现以下的warnning信息&#xff1a; 一些 NuGet 包是使用不同于当前目标框架的目标框架安装的&#…