C/C++ 使用API实现数据压缩与解压缩
在Windows编程中,经常会遇到需要对数据进行压缩和解压缩的情况,数据压缩是一种常见的优化手段,能够减小数据的存储空间并提高传输效率。Windows提供了这些API函数,本文将深入探讨使用Windows API进行数据压缩与解压缩的过程,主要使用ntdll.dll库中的相关函数。
RtlGetCompressionWorkSpaceSize
RtlGetCompressionWorkSpaceSize 函数,位于ntdll.dll库中。该函数用于获取数据压缩所需的工作空间大小。CompressionFormatAndEngine参数指定压缩格式和引擎,CompressBufferWorkSpaceSize和CompressFragmentWorkSpaceSize分别用于输出缓冲区和片段的工作空间大小。
以下是该函数的声明:
typedef NTSTATUS(WINAPI *typedef_RtlGetCompressionWorkSpaceSize)(_In_ USHORT CompressionFormatAndEngine,_Out_ PULONG CompressBufferWorkSpaceSize,_Out_ PULONG CompressFragmentWorkSpaceSize
);
该函数有以下参数:
- CompressionFormatAndEngine:指定压缩格式和引擎的参数。
- CompressBufferWorkSpaceSize:用于输出压缩缓冲区工作空间大小的指针。
- CompressFragmentWorkSpaceSize:用于输出压缩片段工作空间大小的指针。
函数返回NTSTATUS类型的状态码,其中STATUS_SUCCESS表示成功执行。
在使用这个函数时,你需要提供足够大的缓冲区来存储工作空间大小。可以按照以下步骤使用该函数:
- 加载 ntdll.dll 库。
- 获取 RtlGetCompressionWorkSpaceSize 函数地址。
- 定义变量用于存储工作空间大小。
- 调用 RtlGetCompressionWorkSpaceSize 函数,获取工作空间大小。
RtlCompressBuffer
RtlCompressBuffer 同样位于ntdll.dll库中。该函数用于将数据进行压缩。CompressionFormatAndEngine参数指定压缩格式和引擎,UncompressedBuffer和UncompressedBufferSize表示输入的未压缩数据,CompressedBuffer和CompressedBufferSize表示输出的压缩数据,UncompressedChunkSize表示未压缩数据的块大小,FinalCompressedSize表示最终压缩后的大小,WorkSpace表示用于工作的缓冲区。
以下是该函数的声明:
typedef NTSTATUS(WINAPI *typedef_RtlCompressBuffer)(_In_ USHORT CompressionFormatAndEngine,_In_ PUCHAR UncompressedBuffer,_In_ ULONG UncompressedBufferSize,_Out_ PUCHAR CompressedBuffer,_In_ ULONG CompressedBufferSize,_In_ ULONG UncompressedChunkSize,_Out_ PULONG FinalCompressedSize,_In_ PVOID WorkSpace
);
该函数的参数包括:
- CompressionFormatAndEngine:指定压缩格式和引擎的参数。
- UncompressedBuffer:指向待压缩数据的指针。
- UncompressedBufferSize:待压缩数据的大小。
- CompressedBuffer:指向存储压缩数据的缓冲区的指针。
- CompressedBufferSize:存储压缩数据的缓冲区的大小。
- UncompressedChunkSize:未压缩的数据块的大小。
- FinalCompressedSize:用于输出最终压缩数据的大小的指针。
- WorkSpace:用于提供工作空间的指针。
函数返回NTSTATUS类型的状态码,其中STATUS_SUCCESS表示成功执行。
在使用这个函数时,你需要提供足够大的缓冲区来存储压缩后的数据。可以按照以下步骤使用该函数:
- 加载
ntdll.dll库。 - 获取
RtlCompressBuffer函数地址。 - 定义变量并分配内存用于存储未压缩的数据和压缩后的数据。
- 定义变量用于存储工作空间。
- 调用
RtlCompressBuffer函数,将数据进行压缩。 - 处理压缩后的数据。
RtlDecompressBuffer
RtlDecompressBuffer 同样位于ntdll.dll库中。该函数用于将压缩数据进行解压缩。CompressionFormat参数指定压缩格式,UncompressedBuffer和UncompressedBufferSize表示输出的未压缩数据,CompressedBuffer和CompressedBufferSize表示输入的压缩数据,FinalUncompressedSize表示最终解压缩后的大小。
以下是该函数的声明:
typedef NTSTATUS(WINAPI *typedef_RtlDecompressBuffer)(_In_ USHORT CompressionFormat,_Out_ PUCHAR UncompressedBuffer,_In_ ULONG UncompressedBufferSize,_In_ PUCHAR CompressedBuffer,_In_ ULONG CompressedBufferSize,_Out_ PULONG FinalUncompressedSize
);
该函数的参数包括:
- CompressionFormat:指定解压缩的格式。
- UncompressedBuffer:指向存储解压后数据的缓冲区的指针。
- UncompressedBufferSize:存储解压后数据的缓冲区的大小。
- CompressedBuffer:指向待解压数据的指针。
- CompressedBufferSize:待解压数据的大小。
- FinalUncompressedSize:用于输出最终解压后数据的大小的指针。
函数返回NTSTATUS类型的状态码,其中STATUS_SUCCESS表示成功执行。
在使用这个函数时,你需要提供足够大的缓冲区来存储解压后的数据。可以按照以下步骤使用该函数:
- 加载
ntdll.dll库。 - 获取
RtlDecompressBuffer函数地址。 - 定义变量并分配内存用于存储待解压的数据和解压后的数据。
- 调用
RtlDecompressBuffer函数,将数据进行解压。 - 处理解压后的数据。
// 代码来源 《WINDOWS黑客编程技术详解》
// 作者:甘迪文
#include <Windows.h>
#include <iostream>
#include <windef.h>typedef NTSTATUS(WINAPI *typedef_RtlGetCompressionWorkSpaceSize)(_In_ USHORT CompressionFormatAndEngine,_Out_ PULONG CompressBufferWorkSpaceSize,_Out_ PULONG CompressFragmentWorkSpaceSize);typedef NTSTATUS(WINAPI *typedef_RtlCompressBuffer)(_In_ USHORT CompressionFormatAndEngine,_In_ PUCHAR UncompressedBuffer,_In_ ULONG UncompressedBufferSize,_Out_ PUCHAR CompressedBuffer,_In_ ULONG CompressedBufferSize,_In_ ULONG UncompressedChunkSize,_Out_ PULONG FinalCompressedSize,_In_ PVOID WorkSpace);typedef NTSTATUS(WINAPI *typedef_RtlDecompressBuffer)(_In_ USHORT CompressionFormat,_Out_ PUCHAR UncompressedBuffer,_In_ ULONG UncompressedBufferSize,_In_ PUCHAR CompressedBuffer,_In_ ULONG CompressedBufferSize,_Out_ PULONG FinalUncompressedSize);// 数据压缩
BOOL CompressData(BYTE *pUncompressData, DWORD dwUncompressDataLength, BYTE **ppCompressData, DWORD *pdwCompressDataLength)
{BOOL bRet = FALSE;NTSTATUS status = 0;HMODULE hModule = NULL;typedef_RtlGetCompressionWorkSpaceSize RtlGetCompressionWorkSpaceSize = NULL;typedef_RtlCompressBuffer RtlCompressBuffer = NULL;DWORD dwWorkSpaceSize = 0, dwFragmentWorkSpaceSize = 0;BYTE *pWorkSpace = NULL;BYTE *pCompressData = NULL;DWORD dwCompressDataLength = 4096;DWORD dwFinalCompressSize = 0;do{// 加载 ntdll.dll hModule = ::LoadLibrary("ntdll.dll");if (NULL == hModule){ShowError("LoadLibrary");break;}// 获取 RtlGetCompressionWorkSpaceSize 函数地址RtlGetCompressionWorkSpaceSize = (typedef_RtlGetCompressionWorkSpaceSize)::GetProcAddress(hModule, "RtlGetCompressionWorkSpaceSize");if (NULL == RtlGetCompressionWorkSpaceSize){ShowError("GetProcAddress");break;}// 获取 RtlCompressBuffer 函数地址RtlCompressBuffer = (typedef_RtlCompressBuffer)::GetProcAddress(hModule, "RtlCompressBuffer");if (NULL == RtlCompressBuffer){ShowError("GetProcAddress");break;}// 获取WorkSpqce大小status = RtlGetCompressionWorkSpaceSize(COMPRESSION_FORMAT_LZNT1 | COMPRESSION_ENGINE_STANDARD, &dwWorkSpaceSize, &dwFragmentWorkSpaceSize);if (0 != status){break;}// 申请动态内存pWorkSpace = new BYTE[dwWorkSpaceSize];if (NULL == pWorkSpace){break;}::RtlZeroMemory(pWorkSpace, dwWorkSpaceSize);while (TRUE){// 申请动态内存pCompressData = new BYTE[dwCompressDataLength];if (NULL == pCompressData){break;}::RtlZeroMemory(pCompressData, dwCompressDataLength);// 调用RtlCompressBuffer压缩数据RtlCompressBuffer(COMPRESSION_FORMAT_LZNT1, pUncompressData, dwUncompressDataLength, pCompressData, dwCompressDataLength, 4096, &dwFinalCompressSize, (PVOID)pWorkSpace);if (dwCompressDataLength < dwFinalCompressSize){// 释放内存if (pCompressData){delete[]pCompressData;pCompressData = NULL;}dwCompressDataLength = dwFinalCompressSize;}else{break;}}// 返回*ppCompressData = pCompressData;*pdwCompressDataLength = dwFinalCompressSize;bRet = TRUE;} while (FALSE);// 释放if (pWorkSpace){delete[]pWorkSpace;pWorkSpace = NULL;}if (hModule){::FreeLibrary(hModule);}return bRet;
}// 数据解压缩
BOOL UncompressData(BYTE *pCompressData, DWORD dwCompressDataLength, BYTE **ppUncompressData, DWORD *pdwUncompressDataLength)
{BOOL bRet = FALSE;HMODULE hModule = NULL;typedef_RtlDecompressBuffer RtlDecompressBuffer = NULL;BYTE *pUncompressData = NULL;DWORD dwUncompressDataLength = 4096;DWORD dwFinalUncompressSize = 0;do{// 加载 ntdll.dll hModule = ::LoadLibrary("ntdll.dll");if (NULL == hModule){break;}// 获取 RtlDecompressBuffer 函数地址RtlDecompressBuffer = (typedef_RtlDecompressBuffer)::GetProcAddress(hModule, "RtlDecompressBuffer");if (NULL == RtlDecompressBuffer){break;}while (TRUE){// 申请动态内存pUncompressData = new BYTE[dwUncompressDataLength];if (NULL == pUncompressData){break;}::RtlZeroMemory(pUncompressData, dwUncompressDataLength);// 调用RtlCompressBuffer压缩数据RtlDecompressBuffer(COMPRESSION_FORMAT_LZNT1, pUncompressData, dwUncompressDataLength, pCompressData, dwCompressDataLength, &dwFinalUncompressSize);if (dwUncompressDataLength < dwFinalUncompressSize){// 释放内存if (pUncompressData){delete[]pUncompressData;pUncompressData = NULL;}dwUncompressDataLength = dwFinalUncompressSize;}else{break;}}// 返回*ppUncompressData = pUncompressData;*pdwUncompressDataLength = dwFinalUncompressSize;bRet = TRUE;} while (FALSE);// 释放if (hModule){::FreeLibrary(hModule);}return bRet;
}int main(int argc, char *argv[])
{DWORD i = 0;BOOL bRet = FALSE;char szBuffer[] = "DDDDDDDDDDGGGGGGGGGGGG";DWORD dwBufferLength = ::lstrlen(szBuffer);BYTE *pCompressData = NULL;DWORD dwCompressDataLength = 0;BYTE *pUncompressData = NULL;DWORD dwUncompressDataLength = 0;// 压缩数据CompressData((BYTE *)szBuffer, dwBufferLength, &pCompressData, &dwCompressDataLength);// 解压数据UncompressData(pCompressData, dwCompressDataLength, &pUncompressData, &dwUncompressDataLength);// 显示printf("原数据为:\n");for (i = 0; i < dwBufferLength; i++){printf("%X ", szBuffer[i]);}printf("\n\n压缩数据为:\n");for (i = 0; i < dwCompressDataLength; i++){printf("%X ", pCompressData[i]);}printf("\n\n解压缩数据为:\n");for (i = 0; i < dwUncompressDataLength; i++){printf("%X ", pUncompressData[i]);}printf("\n");// 释放if (pUncompressData){delete[]pUncompressData;pUncompressData = NULL;}if (pCompressData){delete[]pCompressData;pCompressData = NULL;}system("pause");return 0;
}
相关文章:
C/C++ 使用API实现数据压缩与解压缩
在Windows编程中,经常会遇到需要对数据进行压缩和解压缩的情况,数据压缩是一种常见的优化手段,能够减小数据的存储空间并提高传输效率。Windows提供了这些API函数,本文将深入探讨使用Windows API进行数据压缩与解压缩的过程&#…...
Visual Studio连接unity编辑器_unity基础开发教程
Visual Studio连接unity编辑器 问题描述解决方法意外情况 问题描述 当我们在unity编辑器中打开C#脚本的时候发现Visual Studio没有连接unity编辑器,在编写代码的时候也没有unity关键字的提醒。 简单来说就是敲代码没有代码提示。 解决方法 这时候需要在unity中进行…...
2023亚太杯数学建模B题思路分析 - 玻璃温室中的微气候法规
1 赛题 问题B 玻璃温室中的微气候法规 温室作物的产量受到各种气候因素的影响,包括温度、湿度和风速[1]。其中,适 宜的温度和风速是植物生长[2]的关键。为了调节玻璃温室内的温度、风速等气候因素 , 温室的设计通常采用带有温室风扇的通风系统…...
轻量封装WebGPU渲染系统示例<37>- 多个局部点光源应用于非金属材质形成的效果(源码)
当前示例源码github地址: https://github.com/vilyLei/voxwebgpu/blob/feature/rendering/src/voxgpu/sample/BasePbrMaterialMultiLights.ts 当前示例运行效果: 此示例基于此渲染系统实现,当前示例TypeScript源码如下: export class BasePbrMaterial…...
设备状态监测与故障诊断系统的作用
随着工业生产的发展和技术的进步,设备状态监测与故障诊断系统在工业领域中扮演着越来越重要的角色。这一系统通过实时监测设备的状态和参数,及时发现潜在的故障,并提供预警信号,以降低生产中断、提高安全性和维护效率。以下将详细…...
浮点数运算精度丢失,如何解决
为什么浮点数运算的时候会有精度丢失的风险? 浮点数运算精度丢失代码演示: float a 2.0f - 1.9f; float b 1.8f - 1.7f; System.out.println(a);// 0.100000024 System.out.println(b);// 0.099999905 System.out.println(a b);// false为什么会出现…...
使用微信小程序openMapApp接口,报错问题解决openMapApp:fail invaild coord
使用微信小程序的 openMapApp 接口时遇到了坐标无效的错误 (openMapApp:fail invalid coord)。这个错误通常是由于提供的地理坐标不符合预期的格式或范围而引起的: 坐标格式: 确保提供的坐标符合正确的格式。常见的格式是 "纬度,经度"…...
2023亚太杯数学建模思路 - 案例:粒子群算法
文章目录 1 什么是粒子群算法?2 举个例子3 还是一个例子算法流程算法实现建模资料 # 0 赛题思路 (赛题出来以后第一时间在CSDN分享) https://blog.csdn.net/dc_sinor?typeblog 1 什么是粒子群算法? 粒子群算法(Pa…...
【开源】基于JAVA的开放实验室管理系统
项目编号: S 013 ,文末获取源码。 \color{red}{项目编号:S013,文末获取源码。} 项目编号:S013,文末获取源码。 目录 一、摘要1.1 项目介绍1.2 项目录屏 二、研究内容2.1 实验室类型模块2.2 实验室模块2.3 实…...
LeetCode48旋转图像
思路是沿对角线交换元素,之后沿矩阵中线交换元素 参考链接 🔗:【LeetCode 每日一题】48. 旋转图像 | 手写图解版思路 代码讲解-哔哩哔哩】 class Solution {public void rotate(int[][] matrix) {int i0,j0;if(matrixnull){return;}int n matrix.length;// int[]…...
sql手工注入漏洞测试(MYSQL)-墨者-url信息
背景: 自己在墨者官网靶场练习的时候,一直出错,手工容易出错,所以列举一些信息供大家核对,可以参考改动。 数据库版本version() 5.7.22-0ubuntu0.16.04.1 当前数据库名称database() m…...
52.seata分布式事务
目录 1.事务的四大特性。 2.分布式服务的事务问题。 3.seata。 3.1理论基础。 3.1.1CAP定理。 3.1.2BASE理论。 3.2初识Seata。 3.2.1Seata的架构。 3.2.2部署TC服务。 3.2.3微服务集成Seata。 3.3 seata提供的四种分布式事务解决方案。 3.3.1 XA模式。 3.3.1.1 X…...
HTML所有功能大汇总
HTML所有的功能,都在下面的表格中呈现清楚了。千万不要死记硬背,但是在遇到困难的时候,可以按照这个表进行查找。 类别功能HTML标签文本样式粗体<b></b> 或 <strong></strong>斜体<i></i>或<em>&…...
层次分析法--可以帮助你做决策的简单算法
作用 层次分析法是一个多指标的评价算法,主要用来在做决策时,给目标的多个影响因子做权重评分。特别是那些需要主观决策的、或者需要用经验判断的决策方案,例如: 买房子(主观决策)选择旅游地(…...
docker启动链接sqlservr的镜像时报SSl错误
本次部署中遇到了老项目中的 net core 5.0 docker 镜像访问sql server 出现SSL Handshake failed with OpenSSL error - SSL_ERROR_SSL.问题 Microsoft.Data.SqlClient.SqlException (0x80131904): A connection was successfully established with the server, but then an e…...
力扣:175. 组合两个表(Python3)
题目: 表: Person ---------------------- | 列名 | 类型 | ---------------------- | PersonId | int | | FirstName | varchar | | LastName | varchar | ---------------------- personId 是该表的主键(具有唯一值的列&#…...
【libGDX】使用Mesh绘制矩形
1 前言 使用Mesh绘制三角形 中介绍了绘制三角形的方法,本文将介绍绘制正方形的方法。 libGDX 以点、线段、三角形为图元,没有提供绘制矩形内部的接口。要绘制矩形内部,必须通过三角形拼接而成,如下图,是通过GL_TRIANGL…...
X2Keyarch迁移工具实战 | 将CentOS高效迁移至浪潮云峦操作系统KeyarchOS
X2Keyarch迁移工具实战 | 将CentOS高效迁移至浪潮云峦操作系统KeyarchOS 1. 搭建仿真线上业务环境2. 安装KeyarchOS操作系统和X2Keyarch迁移工具3. 将CentOS系统业务迁移至KeyarchOS系统 浪潮信息云峦操作系统KeyarchOS基于Linux Kernel、OpenAnolis等开源技术自主研发的一款服…...
基于VM虚拟机下Ubuntu18.04系统,Hadoop的安装与详细配置
参考博客: https://blog.csdn.net/duchenlong/article/details/114597944 与上面这个博客几乎差不多,就是java环境配置以及后面的hadoop的hdfs-site.xml文件有一些不同的地方。 准备工作 1.更新 # 更新 sudo apt update sudo apt upgrade2.关闭防火…...
【图像分类】基于深度学习的垃圾分类系统的设计与实现(ResNet网络,附代码和数据集)
写在前面: 首先感谢兄弟们的关注和订阅,让我有创作的动力,在创作过程我会尽最大能力,保证作品的质量,如果有问题,可以私信我,让我们携手共进,共创辉煌。(专栏订阅用户订阅专栏后免费提供数据集和源码一份,超级VIP用户不在服务范围之内,不想订阅专栏的兄弟们可以私信…...
【Linux 网络】理解并应用应用层协议
HTTP的请求与响应格式 请求格式 请求行:请求方法(GET) 空格 URI(资源地址) 空格 HTTP版本(HTTP/1.1) 换行符(\r\n) 请求报头:表示请求的属性。冒号分割的键值对;每组属性之间使用 \r\n 分隔,遇到空行表示Header部分结束。 请求…...
QMCDecode:macOS上QQ音乐加密音频的终极免费转换方案
QMCDecode:macOS上QQ音乐加密音频的终极免费转换方案 【免费下载链接】QMCDecode QQ音乐QMC格式转换为普通格式(qmcflac转flac,qmc0,qmc3转mp3, mflac,mflac0等转flac),仅支持macOS,可自动识别到QQ音乐下载目录,默认转…...
离散扩散语言模型原理与工程实践
1. 离散扩散语言模型的核心原理与创新价值离散扩散语言模型(Discrete Diffusion Language Models, DLMs)代表了生成式AI领域的最新突破,其核心思想是通过模拟物理扩散过程来实现文本生成。与传统自回归语言模型(ALMs)逐…...
axios 的 GET 请求里,手动写 Content-Type: application/json 基本都会被删掉不是你写法错了是 axios 源码故意这么做的
一、为什么 GET 的 Content-Type 会被删掉看 axios 源码(xhr.js)里的逻辑:if (typeof requestData undefined && key.toLowerCase() content-type) {// Remove Content-Type if data is undefineddelete requestHeaders[key]; }GET…...
别再死磕RPN了!用AI辅助工具快速上手DFMEA的AP(行动优先级)实战
别再死磕RPN了!用AI辅助工具快速上手DFMEA的AP(行动优先级)实战 在汽车和医疗器械行业,设计失效模式与影响分析(DFMEA)是确保产品可靠性的核心工具。然而,许多工程师和质量经理仍在使用传统的风…...
开源AI知识库Tome:基于大语言模型与向量数据库的智能笔记系统
1. 项目概述:当AI遇上知识管理,一个开源智能笔记本的诞生如果你和我一样,每天被海量的信息淹没——浏览器标签页开了一堆,微信收藏夹塞满了文章,笔记软件里躺着无数个“稍后阅读”的链接,最后却什么也没记住…...
用LLaMA-Factory微调ChatGLM3-6B,打造你的专属客服机器人(附数据集模板)
基于LLaMA-Factory微调ChatGLM3-6B构建行业专属客服系统的全流程实践 当电商平台的用户咨询量在促销季激增300%时,传统客服系统往往面临崩溃边缘。去年双十一期间,某头部电商通过定制化AI客服分流了72%的常规咨询,这就是大模型微调技术创造的…...
kodustech/cli:模块化命令行工具集的设计哲学与工程实践
1. 项目概述:一个面向开发者的现代化命令行工具集如果你和我一样,每天的工作都离不开终端,那你肯定对命令行工具又爱又恨。爱的是它的高效和强大,一个命令就能完成图形界面里需要点半天鼠标的操作;恨的是,不…...
你知道吗?其实这些都是AI——物流优化系统
物流优化系统 背景介绍 在全球化经济和电子商务快速发展的背景下,物流成为了商业运作的重要环节。高效的物流系统不仅能够降低企业成本,还能提高客户满意度。然而,传统的物流管理方式通常依赖于人工调度和经验决策,难以应对复杂多变的物流需求和庞大的数据处理量。现代科…...
GTAIV.EFLC.FusionFix季节性事件与彩蛋:探索隐藏的游戏内容
GTAIV.EFLC.FusionFix季节性事件与彩蛋:探索隐藏的游戏内容 【免费下载链接】GTAIV.EFLC.FusionFix This project aims to fix or address some issues in Grand Theft Auto IV: The Complete Edition 项目地址: https://gitcode.com/gh_mirrors/gt/GTAIV.EFLC.Fu…...
