当前位置: 首页 > news >正文

使用Zlib库进行多文件或者多文件夹的压缩解压缩

zlib库可在git上自己clone下来然后使用cmake工具生成解决方案,编译、生成zlib二进制文件。然后将zlib库引入项目:

//zlib库支持
#include "../zlib/include/zlib.h"
#ifdef _DEBUG
#pragma comment(lib, "../zlib/lib/zlibd.lib")
#else
#pragma comment(lib, "../zlib/lib/zlib.lib")
#endif

定义一个文件结构

typedef struct tagZipperFileInfo
{char m_szLocalPath[MAX_PATH];char m_szRootPath[MAX_PATH];char m_szFileName[MAX_PATH];size_t m_FileSize;
}ZipperFileInfo;

压缩文件相关:

/*
*	strFolder	需要被压缩的文件夹
*	strOut		保存的文件
*/
void CompressFolder(std::string& strFolder, std::string& strOut)
{//创建压缩的目标文件std::ofstream dest(strOut, std::ios::binary | std::ios::trunc);if (!dest.is_open()) {//errorreturn;}dest.close();std::vector<ZipperFileInfo> vecZipperFiles;//遍历文件夹下的所有文件OperateFolder(strFolder, strFolder, vecZipperFiles);//压缩文件gzFile gzOut = gzopen(strOut.c_str(), "wb");CompressFiles(vecZipperFiles, gzOut);gzclose(gzOut);
}
void OperateFolder(std::string& strFolder, std::string& strRoot, std::vector<ZipperFileInfo>& vecZipperFiles)
{std::string searchPath = strFolder + "\\*";WIN32_FIND_DATAA findData;HANDLE hFind = FindFirstFileA(searchPath.c_str(), &findData);if (hFind == INVALID_HANDLE_VALUE) {//errorreturn;}do {if (findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) {if (strcmp(findData.cFileName, ".") != 0 && strcmp(findData.cFileName, "..") != 0) {std::string subFolderPath = strFolder + "\\" + findData.cFileName;OperateFolder(subFolderPath, strRoot, vecZipperFiles);}}else {std::string strLocalPath = strFolder + "\\" + findData.cFileName;std::string strFileRootPath = strFolder;std::string strFileName = findData.cFileName;//需要根据strRoot分割出来需要压缩文件的相对路径名std::string strTempPath;size_t start = strFileRootPath.find(strRoot);if (start == std::string::npos)strTempPath = strFileRootPath; // 如果找不到根路径,则返回完整路径else{if (strFileRootPath == strRoot)strTempPath = strFileRootPath.substr(start + strRoot.length());elsestrTempPath = strFileRootPath.substr(start + strRoot.length() + 1);}ZipperFileInfo zipperFile;memcpy(zipperFile.m_szRootPath, strTempPath.c_str(), MAX_PATH);memcpy(zipperFile.m_szLocalPath, strLocalPath.c_str(), MAX_PATH);memcpy(zipperFile.m_szFileName, strFileName.c_str(), MAX_PATH);//计算文件大小std::ifstream in(strLocalPath, std::ios::binary | std::ios::ate);size_t fileSize = in.tellg();in.seekg(0);in.close();zipperFile.m_FileSize = fileSize;vecZipperFiles.push_back(zipperFile);}} while (FindNextFileA(hFind, &findData) != 0);FindClose(hFind);
}

压缩文件:

void CompressFiles(std::vector<ZipperFileInfo>& vecZipperFiles, gzFile& gzOut)
{int nFileCount = vecZipperFiles.size();gzwrite(gzOut, reinterpret_cast<const void*>(&nFileCount), 4);gzwrite(gzOut, reinterpret_cast<const void*>(vecZipperFiles.data()), vecZipperFiles.size() * sizeof(ZipperFileInfo));//再往压缩文件中写入需要压缩为文件内容for (int i = 0; i < vecZipperFiles.size(); i++){std::string strLocalPath = vecZipperFiles[i].m_szLocalPath;std::ifstream infile(strLocalPath, std::ios::binary);char buffer[4096];while (infile){infile.read(buffer, sizeof(buffer));auto bytes = infile.gcount();if (bytes > 0){//写入目标压缩文件gzwrite(gzOut, buffer, bytes);}}infile.close();}
}

文件解压相关:

//strFilePath 压缩文件路径
void DecompressFiles(std::string& strFilePath)
{gzFile gzin = gzopen(strFilePath.c_str(), "rb");if (!gzin) return; //open errorint nFileCount = 0;gzread(gzin, &nFileCount, 4); //读取压缩的文件数量// 读取文件列表信息std::vector<ZipperFileInfo> vecZipperFiles;ZipperFileInfo zipperFile;for (int i = 0; i < nFileCount; i++){if (gzread(gzin, &zipperFile, sizeof(ZipperFileInfo)) == sizeof(ZipperFileInfo))vecZipperFiles.push_back(zipperFile);}SStringW sstrAppPath = CGlobalUnits::GetInstance()->m_sstrAppPath;std::string strAppPath = S_CW2A(sstrAppPath);//先创建个输出目录size_t szPos = strFilePath.find_last_of("\\");if (szPos != std::string::npos){std::string strTmp = strFilePath.substr(szPos + 1);//分解出namesize_t szName = strTmp.find_last_of(".");if (szName != std::string::npos){std::string strName = strTmp.substr(0, szName);strAppPath += strName;}}CreateDirectoryA(strAppPath.c_str(), NULL);//解压文件for (int i = 0; i < vecZipperFiles.size(); i++){ZipperFileInfo& info = vecZipperFiles[i];std::string strRoot = info.m_szRootPath;std::string strPath;if (strRoot == "")  //根目录下strPath = strAppPath + "\\" + info.m_szFileName;else{CreateFolder(strAppPath, strRoot);strPath = strAppPath + "\\" + strRoot + "\\" + info.m_szFileName;}std::ofstream outFile(strPath, std::ios::binary);if (!outFile) continue;  //error char buffer[4096];size_t fileSize = info.m_FileSize;while (fileSize > 0){size_t bytesToRead = std::min(static_cast<size_t>(4096), fileSize);int bytesRead = gzread(gzin, buffer, bytesToRead);if (bytesRead <= 0) break; //read erroroutFile.write(buffer, bytesRead);fileSize -= bytesRead;}outFile.close();}gzclose(gzin);
}

递归生成压缩文件中的目录结构:

/*
*	strRoot	解压缩的目标目录
*	strDir	压缩文件的相对路径
*/
void CreateFolder(std::string& strRoot, std::string& strDir)
{size_t szPos = strDir.find_first_of("\\");if (szPos != std::string::npos){std::string strName = strDir.substr(0, szPos);std::string strPath = strRoot + "\\" + strName;CreateDirectoryA(strPath.c_str(), NULL);std::string strSubName = strDir.substr(szPos + 1);std::string strTempRoot = strRoot + "\\" + strName;CreateFolder(strTempRoot, strSubName);}else{std::string strPath = strRoot + "\\" + strDir;CreateDirectoryA(strPath.c_str(), NULL);}
}

相关文章:

使用Zlib库进行多文件或者多文件夹的压缩解压缩

zlib库可在git上自己clone下来然后使用cmake工具生成解决方案&#xff0c;编译、生成zlib二进制文件。然后将zlib库引入项目&#xff1a; //zlib库支持 #include "../zlib/include/zlib.h" #ifdef _DEBUG #pragma comment(lib, "../zlib/lib/zlibd.lib") …...

CSGHub携手Nvidia NIM、阿里计算巢打造企业级私有化部署解决方案

强强联合 人工智能与大数据的迅速发展&#xff0c;大模型的推理应用和资产管理已成为企业数字化转型的重要组成部分&#xff0c;企业正寻求高效、安全的AI模型部署解决方案。为应对日益增长的计算需求和复杂的数据管理挑战&#xff0c;CSGHub、Nvidia和阿里云计算巢强强联手&a…...

opencv的球面投影

cv::detail::SphericalProjector 在全景图像拼接任务中&#xff0c;可能需要对多个图像进行球面投影以实现无缝拼接。每个cv::detail::SphericalProjector可以负责一个图像的球面投影操作。通过将多个这样的投影器存储在std::vector中&#xff0c;可以对一组图像依次进行投影处…...

5. 去中心化应用(dApp)

去中心化应用&#xff08;dApp&#xff09; 去中心化应用&#xff08;dApp&#xff09;是基于区块链技术构建的应用程序&#xff0c;其核心特性是去中心化、透明和开放。dApp与传统应用有许多显著的区别&#xff0c;它们在实现和功能上都带来了新的变革。以下是对dApp的详细介…...

k8s服务发布Ingress

Kubernetes暴露服务的方式目前只有三种&#xff1a;LoadBlancer Service、NodePort Service、Ingress&#xff0c;通俗来讲&#xff0c;ingress和之前提到的Service、Deployment&#xff0c;也是一个k8s的资源类型&#xff0c;ingress用于实现用域名的方式访问k8s内部应用。 In…...

区块链学习笔记1--比特币

区块链是分布式数据存储、点对点传输、共识机制、加密算法等计算机技术的新型应用模式。 从狭义上来说&#xff1a;区块链是一种按照时间顺序将数据区块以顺序相连的方式组合成的一种链式数据结构&#xff0c;并以密码学的方式保证的不可篡改和不可伪造的分布式账本。 意思就是…...

在 Vite 项目中自动为每个 Vue 文件导入 base.less

在 Vue.js 项目中&#xff0c;使用 Less 作为 CSS 预处理器时&#xff0c;我们通常会创建一个全局的样式文件&#xff08;如 base.less&#xff09;&#xff0c;用于存放一些全局变量、混合、通用样式等。为了避免在每个 Vue 组件中手动导入这个文件&#xff0c;我们可以通过配…...

RUST 学习之全局变量

RUST 全局变量 rust 全局变量编译期初始化的全局变量静态常量静态变量原子类型的静态变量 运行期初始化的全局变量lazy_staticBox::leakOnceCell & OnceLock 参考文档 rust 全局变量 编译期初始化的全局变量 静态常量 在编译期初始化&#xff0c;所以其赋值只能是表达式…...

代码随想录八股训练营第三十九天| C++

前言 一、说一下 lambda函数&#xff1f; 1.1.Lambda 函数的一般语法如下: 1.2.捕获子句&#xff1a; 二、C 怎么实现一个单例模式&#xff1f; 2.1.懒汉式&#xff08;线程不安全&#xff09;: 2.2.饿汉式&#xff08;线程安全&#xff09;: 2.3.双重检查锁定&#xff…...

服务网关工作原理,如何获取用户真实IP?

文章目录 一、什么是网关二、网关工作原理 (★)三、SpringCloud Gateway3.1 Gateway 简介3.2 Gateway 环境搭建3.3 自定义路由规则 (★)3.4 局部过滤器3.5 全局过滤器&#xff08;案例&#xff1a;获取用户真实IP地址&#xff09; (★) 补充1&#xff1a;不同类型的客户端如何设…...

单链表的实现(C语言)

目录 1.单链表 1.1 实现单链表 1.1.1 文件创建 1.1.2 链表功能了解 1.1.3 链表的结点 1.1.4 链表的函数声明 1.1.5 链表功能的实现 链表是一种链式结构&#xff0c;物理结构不连续&#xff0c;逻辑结构是连续的&#xff0c;在计算机中链表的实际存储是按照一个结点内存放…...

sql语句的训练2024/9/9

1题 需要看清思路&#xff1a;不是将数据库中的device_id的名字改为user_infors_example&#xff0c;而是在查找的时候&#xff0c;需要将device_id看成user_infors_example来进行查找。 答案 select device_id AS user_infos_example FROM user_profile limit 2 2 当固定查找…...

【QT】常用控件-下

欢迎来到Cefler的博客&#x1f601; &#x1f54c;博客主页&#xff1a;折纸花满衣 &#x1f3e0;个人专栏&#xff1a;QT 目录 &#x1f449;&#x1f3fb;QComboBox&#x1f449;&#x1f3fb; QSpinBox&#x1f449;&#x1f3fb;QDateTimeEdit&#x1f449;&#x1f3fb;QD…...

828华为云征文|华为云Flexus X实例docker部署Jitsi构建属于自己的音视频会议系统

828华为云征文&#xff5c;华为云Flexus X实例docker部署Jitsi构建属于自己的音视频会议系统 华为云最近正在举办828 B2B企业节&#xff0c;Flexus X实例的促销力度非常大&#xff0c;特别适合那些对算力性能有高要求的小伙伴。如果你有自建MySQL、Redis、Nginx等服务的需求&a…...

25虾皮笔试shopee笔试测评sea笔试测评题型

虾皮笔试shopee笔试测评用的自己的笔试系统&#xff0c;全英文笔试&#xff1a; 1.Numerical Reasoning Test&#xff1a;10题&#xff0c;言语推断和数学计算 2. Verbal Reasoning Test&#xff1a;10题&#xff0c;言语理解&#xff0c;每题一段英文材料&#xff0c;选对错…...

启明云端乐鑫代理商,乐鑫ESP32无线芯片方案,物联网设备WiFi联动控制

随着智能和远程技术的飞速发展&#xff0c;物联网(IoT)逐渐出现在我们生活的每一个角落。乐鑫以其创新的无线通信技术&#xff0c;正成为智能家居、工业自动化和医疗设备等领域的推动者。 无线WiFi芯片模组不仅提供了强大的数据处理能力&#xff0c;还赋予了设备以直观的交互方…...

希尔排序/选择排序

前言&#xff1a; 本篇主要对常见的排序算法进行简要分析&#xff0c;代码中均以数组 arr[] { 5, 3, 9, 6, 2, 4, 7, 1, 8 } 为例&#xff0c;进行升序排列。 常见的排序算法有如下&#xff1a; 选择排序中&#xff0c;直接选择排序没有任何实际与教育意义&#xff0c;而堆排…...

漫谈设计模式 [16]:中介者模式

引导性开场 菜鸟&#xff1a;老鸟&#xff0c;我最近在开发一个聊天应用的时候遇到了点问题。每个用户都需要与其他用户直接通信&#xff0c;这让我在代码中写了很多复杂的逻辑来管理这些联系。这样下去&#xff0c;代码越来越难维护了。你有什么建议吗&#xff1f; 老鸟&…...

深度学习-物体检测YOLO(You only look once)

目录 一&#xff1a;YOLO算法的网络结构 流程 1.图像分割 2.图片在网格中的处理 3.非极大值抑制 二&#xff1a;训练 三&#xff1a;分类误差 四&#xff1a;与Faster R-CNN对比 一&#xff1a;YOLO算法的网络结构 GooleNet4个卷积2个全连接层 流程 输入原始图片resize到…...

redisson中的分布式锁

我的博客大纲 我的后端学习大纲 a.redisson概述&#xff1a; 1.Redisson是一个在Redis的基础上实现的Java驻内存数据网格&#xff08;In-Memory Data Grid&#xff09;2.redisson介绍官方文档地址&#xff1a;3.Redisson它不仅提供了一系列的分布式的Java常用对象&#xff0c;还…...

避坑指南:Houdini风格化树木导入Unity URP后,光照和裁剪效果不对怎么办?

Houdini风格化树木在Unity URP中的渲染问题深度解析与实战修复 当你在Houdini中精心雕琢的风格化树木模型导入Unity URP管线后&#xff0c;可能会遭遇一系列令人沮丧的渲染问题&#xff1a;叶片边缘出现锯齿状裁剪、光照效果与预期不符、阴影投射异常等。这些问题的根源往往在于…...

LAV Filters深度解析:开源DirectShow媒体解码器的架构原理与性能优化指南

LAV Filters深度解析&#xff1a;开源DirectShow媒体解码器的架构原理与性能优化指南 【免费下载链接】LAVFilters LAV Filters - Open-Source DirectShow Media Splitter and Decoders 项目地址: https://gitcode.com/gh_mirrors/la/LAVFilters LAV Filters是一套基于F…...

Windows 11终极优化指南:使用Win11Debloat实现专业级系统调校

Windows 11终极优化指南&#xff1a;使用Win11Debloat实现专业级系统调校 【免费下载链接】Win11Debloat A simple, lightweight PowerShell script that allows you to remove pre-installed apps, disable telemetry, as well as perform various other changes to declutter…...

CATCCOS核心组件深度解析:从Host到Device的分层架构设计原理

CATCCOS核心组件深度解析&#xff1a;从Host到Device的分层架构设计原理 【免费下载链接】catccos CATCCOS昇腾计算-通信融合算子模板库&#xff0c;是一个聚焦于提供高性能计算通信融合类算子基础模板的代码库。 项目地址: https://gitcode.com/cann/catccos CATCCOS昇…...

金融公共服务机构钓鱼邮件威胁治理研究 —— 以 NSI 安全事件为例

摘要 英国国家储蓄与投资机构 NS&I 近三年拦截各类恶意邮件 132,126 封&#xff0c;其中垃圾邮件 97,777 封&#xff0c;钓鱼攻击从 1,043 起激增至 4,414 起&#xff0c;呈现总量下降但精准化、AI 化、高危害性显著上升的趋势。作为管理海量公众资金与敏感数据的金融公共服…...

如何一键自动化部署Office:LKY Office Tools完整配置指南

如何一键自动化部署Office&#xff1a;LKY Office Tools完整配置指南 【免费下载链接】LKY_OfficeTools 一键自动化 下载、安装、激活 Office 的利器。 项目地址: https://gitcode.com/GitHub_Trending/lk/LKY_OfficeTools 在Windows系统中安装Microsoft Office一直是个…...

RAG知识库全流程实操:从分块→检索→生成,逐步拆解

搭了个 RAG&#xff0c;文档灌进去&#xff0c;问题丢过来&#xff0c;回答出来了——看起来能用了。 但问它"RAG 四代架构是什么"&#xff0c;它编了个"第一代 RTG"——这个术语根本不存在。问它"嵌入模型中文怎么选"&#xff0c;它说"建…...

机器人企业如何用 CRM 优化线索、商机与客户管理

对于机器人、工业自动化和智能制造解决方案企业而言&#xff0c;销售管理往往不是简单的客户跟进&#xff0c;而是围绕复杂需求、技术方案、项目周期和多角色协作展开的长期过程。Zoho CRM 的价值&#xff0c;正是在于帮助这类 B2B 企业把线索管理、商机推进、客户需求沉淀和销…...

苹果Siri 2025全面升级:从LLM集成到系统级智能体的技术路径解析

1. 项目概述&#xff1a;一次迟来的“大脑”升级最近&#xff0c;关于苹果Siri将在2025年迎来全面优化的消息&#xff0c;在圈内引发了不小的讨论。作为一名长期关注智能交互领域发展的从业者&#xff0c;我对此并不感到意外&#xff0c;反而觉得这是一次“虽迟但到”的必要手术…...

快速上手3DGS数字孪生开发:一份必做的技术动作盘点清单

一、行业核心技术科普&#xff1a;3DGS数字孪生开发的关键技术节点从零开始构建一个基于3D高斯泼溅&#xff08;3DGS&#xff09;的数字孪生应用&#xff0c;涉及多个关键技术节点。每个节点的执行质量&#xff0c;都直接影响最终应用的性能与用户体验。其域创新推出的LCC格式&…...