c++ 解析zip文件,实现对流式文件pptx内容的修改

libzip

  • 官网地址:
  • 示例代码
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <ctime>
#include <zip.h>//解析原始zip内容,保存为新的zip文件
int ziptest(const char* inputPath, const char* outputPath)
{int error = 0;zip_t *zip_file = zip_open(inputPath, ZIP_CHECKCONS, &error);if (zip_file == NULL) {printf("Failed to open zip file: %s\n", zip_strerror(zip_file));return 1;}// 获取条目数量int numEntries = zip_get_num_entries(zip_file, 0);if (numEntries < 0) {std::cerr << "Failed to get number of entries." << std::endl;zip_close(zip_file);return 1;}//保存为目标zip文件zip_t* archive = zip_open(outputPath, ZIP_CREATE | ZIP_TRUNCATE, nullptr);if (archive == nullptr) {std::cout << "无法创建 ZIP 存档." << std::endl;return 1;}unsigned char* itemData = NULL;for (size_t i = 0; i < numEntries; i++) {zip_stat_t entryStat;if (zip_stat_index(zip_file, i, 0, &entryStat) != 0) {std::cerr << "Failed to get information for entry " << i << std::endl;continue;}//printf("index: [%llu]\t", entryStat.index);//printf("Name: [%s]\t", entryStat.name);//printf("valid: [%llu]\t", entryStat.valid);//printf("Size: [%llu]\t", entryStat.size);//printf("comp_size: [%llu]\t", entryStat.comp_size);//printf("comp_method: [%zu]", entryStat.comp_method);//printf("\t flags: [%lu]\n", entryStat.flags);if (entryStat.valid & ZIP_STAT_NAME) {// 打开条目文件zip_file_t *entryFile = zip_fopen_index(zip_file, i, 0);if (entryFile == NULL) {std::cerr << "Failed to open entry file: " << entryStat.name << std::endl;//goto END;continue;}size_t bufferSize = entryStat.size;// 读取内存空间itemData = (unsigned char *)malloc(bufferSize);if (zip_fread(entryFile, itemData, bufferSize) < 0) {std::cerr << "Failed to read entry file: " << entryStat.name << std::endl;zip_fclose(entryFile);if (itemData != NULL) {free(itemData);itemData = NULL;}continue;}// 创建源对象,并将其添加到 ZIP 存档中//zip_source_buffer内部会自动释放itemData内存zip_source* source = zip_source_buffer(archive, itemData, bufferSize, 0);if (source == NULL) {std::cout << "无法写入 ZIP 文件." << std::endl;if (itemData != NULL) {free(itemData);itemData = NULL;}zip_fclose(entryFile);continue;}//  use zip_file_replace() to modify source zip file/****if (zip_file_replace(zip_file, i, source, 0) < 0) {std::cout << "replace failed." << std::endl;zip_source_free(source);zip_fclose(entryFile);break;}****/if (zip_file_add(archive, entryStat.name, source, ZIP_FL_OVERWRITE) < 0) {std::cout << "无法写入 ZIP 文件." << std::endl;zip_source_free(source);zip_fclose(entryFile);if (itemData != NULL) {free(itemData);itemData = NULL;}continue;}zip_fclose(entryFile);}}// 关闭zip文件zip_close(zip_file);zip_close(archive);return 0;
}
#include <zip.h>
#include <memory.h>
#include <stdio.h>
#include <math.h>
#include <fstream>#ifdef _WIN32
#include <io.h> /* _access */
#include<direct.h> /* _mkdir */
#include<windows.h>
#else
#include<unistd.h> /* access */
#include<sys/stat.h> /*mkdir*/
#include <sys/types.h>
#endif //创建多级文件夹
void createFolders(std::string rootPath);
//获取多级文件夹下所以文件列表
void getDirAllFilePath(const char* folderPath, std::vector<std::string>& filePaths);//解压zip,保存到磁盘指定目录
int unzipFunc(const char* destzip, const std::string output) {// 打开ZIP文件zip* archive = zip_open(destzip , 0, NULL);if (!archive) {std::cerr << "Failed to open archive" << std::endl;return -1;}// 获取ZIP文件中的文件数量int numFiles = zip_get_num_files(archive);//std::cout << "Archive contains " << numFiles << " files" << std::endl;// 遍历ZIP文件中的所有文件for (int i = 0; i < numFiles; ++i) {FILE *fp = NULL;// 获取文件的名称和大小zip_stat_t fileStat;zip_stat_init(&fileStat);if (zip_stat_index(archive, i, 0, &fileStat) != 0) {std::cerr << "Failed to get file info for index " << i << std::endl;continue;}int len = strlen(fileStat.name);std::cout << "File " << i << ": " << fileStat.name << " (" << fileStat.size << " bytes)" << std::endl;// 解压文件到磁盘指定位置, 执行解压之前, 需创建对应的文件夹,否则,解压失败std::string dest_name = output + std::string(fileStat.name);createFolders(dest_name);zip_file_t* zf = zip_fopen_index(archive, i, 0);if (!zf) { continue; }fp = fopen(dest_name.c_str(), "wb");if (fp == NULL) { continue; }long long sum = 0;unsigned char* buffer = (unsigned char*)malloc(fileStat.size);memset(buffer, 0, fileStat.size);if (zip_fread(zf, buffer, fileStat.size) < 0) {continue;}fwrite(buffer, 1, fileStat.size, fp);free(buffer);buffer = NULL;zip_fclose(zf);fclose(fp);}// 关闭ZIP文件if (zip_close(archive) != 0) {std::cerr << "Failed to close archive" << std::endl;return -1;}return 0;
}//将文件夹压缩为指定的zip
int zipFunc(const char* inputDirPath, const char* destzip) {// 打开ZIP文件int iErr = 0;zip* archive = zip_open(destzip, ZIP_CREATE | ZIP_TRUNCATE, &iErr);if (!archive) {std::cerr << "Failed to open archive" << std::endl;return -1;}std::vector<std::string> files;//获取的是绝对路径getDirAllFilePath(inputDirPath, files);std::string rootPath(inputDirPath);for (size_t i = 0; i < files.size(); i++) {std::string::size_type rootSize = rootPath.length();//获取指定目录下的相对路径,作为zip包的条目名称std::string itemName = files[i].substr(rootSize + 1);std::cout << "entry: " << files[i].c_str() << std::endl;//通过文件创建zip_source源对象zip_source_t* source = zip_source_file(archive, files[i].c_str(), 0, -1);if (!source){printf(" open zip_source file failed\n");zip_close(archive);return 1;}//add fileif (zip_file_add(archive, itemName .c_str(), source, ZIP_FL_OVERWRITE) < 0) {zip_source_free(source);zip_close(archive);return 2;}}// 关闭ZIP文件if (zip_close(archive) != 0) {std::cerr << "Failed to close archive" << std::endl;return -1;}return 0;
}//创建文件夹
int create_dir(const char *dir)
{
#ifdef WIN32if ((_access(dir, 0)) != 0)	//如果文件夹不存在{int flag = _mkdir(dir);if (flag != 0){printf("Fail to create directory %s.", dir); //"Fail to create directory." << std::endl;return OOXML_PARAMETER_ERR;}}
#elseif ((access(dir, 0)) != 0)	//如果文件夹不存在{int flag = mkdir(dir, S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH);if (flag != 0){printf("Fail to create directory %s.", dir);return OOXML_PARAMETER_ERR;}}
#endifreturn 0;
}
//递归创建文件夹
void mkdirRecursively(const std::string root, const std::string folderPath)
{size_t pos = folderPath.find('/');std::string subPath;if (pos != std::string::npos){subPath = folderPath.substr(0, pos);}else {subPath = folderPath;}if (root == "") {if (!subPath.empty()){if(create_dir(subPath.c_str()) != 0) return ;std::string remainingPath = folderPath.substr(pos + 1);if (pos != std::string::npos) {return 	mkdirRecursively(subPath, remainingPath);}else {return ;}}}else {std::string subdir = root;if (subdir.back() != '/') {subdir += "/";}if (!subPath.empty()){std::string current = subdir + subPath;if (create_dir(current.c_str()) != 0) return ;std::string remainingPath = folderPath.substr(pos + 1);if (pos != std::string::npos) {return 	mkdirRecursively(current, remainingPath);}else {return ;}}}return ;
}//创建多级文件夹
//如果是文件,则创建文件所占目录文件夹
void createFolders(std::string rootPath)
{std::string::size_type idx = rootPath.find_last_of('/');std::string filename = rootPath.substr(idx + 1);std::string::size_type pos = filename.find('.');if (!filename.empty() && (pos != std::string::npos)) {std::string subdirname = rootPath.substr(0, idx + 1);//std::cout << subdirname << std::endl;mkdirRecursively("", subdirname);}else{//std::cout << rootPath << std::endl;mkdirRecursively("", rootPath);}
}void getDirAllFilePath(const char* folderPath, std::vector<std::string>& filePaths)
{
#ifdef  WIN32HANDLE hFind;WIN32_FIND_DATA findData;LARGE_INTEGER size;char dirNew[128] = { 0x0 };// 向目录加通配符,用于搜索第一个文件 strcpy(dirNew, folderPath);strcat(dirNew, "\\*.*");hFind = FindFirstFile(dirNew, &findData);if (hFind == INVALID_HANDLE_VALUE){std::cerr << "无法打开目录:" << folderPath << std::endl;return;}do{// 是否是文件夹,并且名称不为"."或".." //std::cout << findData.dwFileAttributes << "\n";if (findData.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY&& strcmp(findData.cFileName, ".") != 0&& strcmp(findData.cFileName, "..") != 0){// 将dirNew设置为搜索到的目录,并进行下一轮搜索 std::string file(findData.cFileName);std::string newPath = std::string(folderPath) + "/" + file;getFiles(newPath.c_str(), filePaths);}else if (strcmp(findData.cFileName, ".") == 0 || strcmp(findData.cFileName, "..") == 0) {continue;}else {std::string file(findData.cFileName);std::string fullPath = std::string(folderPath) + "/" + file;filePaths.push_back(fullPath);}} while (FindNextFile(hFind, &findData));FindClose(hFind);
#elseDIR* dir = opendir(folderPath);if (dir == nullptr){std::cerr << "无法打开目录:" << folderPath << std::endl;return;}struct dirent* entry;while ((entry = readdir(dir)) != nullptr){std::string entryName(entry->d_name);if (entryName == "." || entryName == ".."){continue;}std::string entryPath(folderPath);if (entryPath.back() != '/') { entryPath += "/"; }entryPath +=entryName;struct stat entryStat;if (stat(entryPath.c_str(), &entryStat) == -1){std::cerr << "无法获取文件属性:" << entryPath << std::endl;continue;}if (S_ISDIR(entryStat.st_mode)){// 如果是子目录,则递归调用该函数获取子文件夹中的文件路径getFiles(entryPath.c_str(), filePaths);}else if (S_ISREG(entryStat.st_mode)){// 如果是文件,则添加到路径列表中filePaths.push_back(entryPath);}}closedir(dir);
#endif
}

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

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

相关文章

vue pc官网顶部导航栏组件

官网顶部导航分为一级导航和二级导航 导航的样子 文件的层级 router 文件层级 header 组件代码 <h1 class"logo-wrap"><router-link to"/"><img class"logo" :src"$config.company.logo" alt"" /><i…

直面双碳目标,优维科技携手奥意建筑打造绿色低碳建筑数智云平台

优维“双碳”战略合作建筑 为落实创新驱动发展战略&#xff0c;增强深圳工程建设领域科技创新能力&#xff0c;促进技术进步、科技成果转化和推广应用&#xff0c;根据《深圳市工程建设领域科技计划项目管理办法》《深圳市住房和建设局关于组织申报2022年深圳市工程建设领域科…

K8S集群优化的可执行优化

目录 前期环境优化 1.永久关闭交换分区 2.#加载 ip_vs 模块 3.调整内核参数 4.#使用Systemd管理的Cgroup来进行资源控制与管理 5.开机自启kubelet 6.内核参数优化方案 7.etcd优化 默认etcd空间配额大小为 2G&#xff0c;超过 2G 将不再写入数据。通过给etcd配置 --quo…

IO流(Java)

IO流 在学习IO流之前&#xff0c;我们首先了解一下File File File即文件或文件夹路径对象&#xff0c;其示例类可以是存在路径也可以是未创造路径 File有什么用 用于创建或操作文件或文件夹 File常用API API部分看得懂会查会用即可 IO流 IO(Input 读数据 Output写数据…

Qt/QML编程学习之心得:工程中的文件(十二)

Qt生成了工程之后,尤其在QtCreator产生对应的project项目之后,就如同VisualStudio一样,会产生相关的工程文件,那么这些工程文件都是做什么的呢?这里介绍一下。比如产生了一个Qt Widget application,当然如果Qt Quick Application工程会有所不同。 一、.pro和.pro.user …

企业计算机服务器中了360勒索病毒如何解密,勒索病毒解密数据恢复

网络技术的不断应用与发展&#xff0c;为企业的生产运营提供了极大便利&#xff0c;但随之而来的网络安全威胁也不断增加。近期&#xff0c;云天数据恢复中心接到很多企业的求助&#xff0c;企业的计算机服务器遭到了360后缀勒索病毒攻击&#xff0c;导致企业的所有数据被加密&…

游戏策划常用的ChatGPT通用提示词模板

游戏设计&#xff1a;请帮助我设计一个有趣的游戏。 游戏玩法&#xff1a;如何设计游戏的玩法&#xff1f; 游戏机制&#xff1a;如何设计游戏的机制&#xff1f; 游戏平衡&#xff1a;如何平衡游戏中的各种元素&#xff1f; 游戏美术&#xff1a;如何设计游戏的美术风格&a…

『PyTorch学习笔记』如何快速下载huggingface模型/数据—全方法总结

如何快速下载huggingface模型/数据—全方法总结 文章目录 一. 如何快速下载huggingface大模型1.1. IDM(Windows)下载安装连接1.2. 推荐 huggingface 镜像站1.3. 管理huggingface_hub cache-system(缓存系统) 二. 参考文献 一. 如何快速下载huggingface大模型 推荐 huggingface…

希亦洗地机跟追觅洗地机入手哪个更好?追觅跟希亦洗地机深度评估

近年来&#xff0c;洗地机可以同时处理干湿垃圾&#xff0c;同时降低用户在清洁过程中的劳动强度&#xff0c;成为了家居清洁的新宠&#xff0c;但是目前市场上的品牌和型号层出不穷。用户往往很难挑选&#xff0c;本文挑选了两款目前口碑最好的两款洗地机给大家做一个全面的评…

Android 记录一些Framework开发的命令

源码编译流程 1. "source build/envsetup.sh" (source可以用 . 代替&#xff0c;即". build/envsetup.sh") 2. "lunch"&#xff0c;并选择要编译的项目或者"choosecombo" 3. "make idegen -j4" (这里的 -j4 表示用4线程来…

LeeCode每日刷题12.8

搜索插入位置 给定一个排序数组和一个目标值&#xff0c;在数组中找到目标值&#xff0c;并返回其索引。如果目标值不存在于数组中&#xff0c;返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 示例 1: 输入: nums [1,3,5,6], target 5 输出: …

外贸行业的CRM系统和其它CRM有什么区别?

外贸行业对客户管理的追求日益提高&#xff0c;为了应对客户需求的变化和多元性&#xff0c;外贸企业需要借助CRM管理系统实现智能管理。下面&#xff0c;我们将详细探讨外贸CRM的概念、特点和具体应用。 什么是外贸CRM&#xff1f; 外贸CRM是指针对外贸行业的客户关系管理系…

Nginx+Promtail+Loki+Grafana 升级ELK强大工具

最近客户有个新需求,就是想查看网站的访问情况,由于网站没有做google的统计和百度的统计,所以访问情况,只能通过日志查看,通过脚本的形式给客户导出也不太实际,给客户写个简单的页面,咱也做不到 成熟的日志解决方案,那就是ELK,还有现在比较火的Loki,(当然还有很多其…

两电脑共享鼠标键盘方案

一开始使用的是shareMouse 但是需要注册还有很多不稳定问题 后来想买个双拷线&#xff0c;又太贵&#xff0c;感觉不值的。 再后来&#xff0c;发现微软有自己的系统上的 共享方案 &#xff0c;叫做 Mouse without Borders ,而且是免费的&#xff0c;只能在window电脑上使用…

Linus:我休假的时候也会带着电脑,否则会感觉很无聊

目录 Linux 内核最新版本动态 关于成为内核维护者 代码好写&#xff0c;人际关系难处理 内核维护者老龄化 内核中 Rust 的使用 关于 AI 的看法 参考 12.5-12.6 日&#xff0c;Linux 基金会组织的开源峰会&#xff08;OSS&#xff0c;Open Source Summit&#xff09;在日…

基于Java酒店管理系统

基于Java酒店管理系统 功能需求 1、客房管理&#xff1a;系统需要提供客房的信息管理功能&#xff0c;包括客房的类型、数量、价格、状态&#xff08;如空闲、已预订、已入住等&#xff09;等信息。此外&#xff0c;系统还应支持对客房信息的修改和查询操作。 2、预订管理&a…

日常开发日志

目录 1、idea开发服务启动的网页地址不显示前端样式&#xff1a; 2、java Date 与myibits 的空判断&#xff1a; 1、idea开发服务启动的网页地址不显示前端样式&#xff1a; idea开发时&#xff0c;tomcat启动的后端弹出的网页地址&#xff0c;呈现的网页没有样式&#xff0…

报名学历的同学,月底前记得申请抵扣个税!

2024年度专项附加扣除开始确认啦&#xff01; 已经报名学历&#xff08;自考、成考、开放大学&#xff09;的同学&#xff0c;记得去申请抵扣个税哦&#xff01; 每个月的应纳税额可以减免400元呢&#xff0c;学历提升在读这几年算下来&#xff0c;可以省不少钱。 注意&#x…

轮播插件Slick.js使用方法详解

相比于Swiper而选择使用Slick.js的原因主要是因为其兼容不错并且在手机端的滑动效果更顺畅 参数&#xff1a; 1.基本使用&#xff1a;一般使用只需前十个属性 $(.box ul).slick({autoplay: true, //是否自动播放pauseOnHover: false, //鼠标悬停暂停自动播放speed: 1500, //…

linux链表应用2

以下是根据您的要求对示例代码进行优化的结果&#xff1a; #include <stdio.h> #include <stdlib.h> #include <string.h>// 定义路径节点结构体 typedef struct PathNode {unsigned int fildid;char folderName[100]; // 文件夹名字struct PathNode* next…