亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

C/C++使用Zlib實(shí)現(xiàn)文件的壓縮與解壓

 更新時(shí)間:2023年11月28日 10:34:36   作者:微軟技術(shù)分享  
zlib 是一個(gè)開(kāi)源的數(shù)據(jù)壓縮庫(kù),旨在提供高效、輕量級(jí)的壓縮和解壓縮算法,本文將介紹如何使用 zlib 庫(kù)進(jìn)行數(shù)據(jù)的壓縮和解壓縮,以及如何保存和讀取壓縮后的文件,感興趣的可以了解下

在軟件開(kāi)發(fā)和數(shù)據(jù)處理中,對(duì)數(shù)據(jù)進(jìn)行高效的壓縮和解壓縮是一項(xiàng)重要的任務(wù)。這不僅有助于減小數(shù)據(jù)在網(wǎng)絡(luò)傳輸和存儲(chǔ)中的占用空間,還能提高系統(tǒng)的性能和響應(yīng)速度。本文將介紹如何使用 zlib 庫(kù)進(jìn)行數(shù)據(jù)的壓縮和解壓縮,以及如何保存和讀取壓縮后的文件。zlib 是一個(gè)開(kāi)源的數(shù)據(jù)壓縮庫(kù),旨在提供高效、輕量級(jí)的壓縮和解壓縮算法。其核心壓縮算法基于 DEFLATE,這是一種無(wú)損數(shù)據(jù)壓縮算法,通常能夠提供相當(dāng)高的壓縮比。zlib 庫(kù)廣泛應(yīng)用于多個(gè)領(lǐng)域,包括網(wǎng)絡(luò)通信、文件壓縮、數(shù)據(jù)庫(kù)系統(tǒng)等。

保存文件

使用 CreateFile 打開(kāi)文件,通過(guò) WriteFile 向文件中寫(xiě)出數(shù)據(jù),最后調(diào)用 CloseHandle 關(guān)閉句柄,實(shí)現(xiàn)文件的保存。

#define ZLIB_WINAPI
#include <string>
#include <iostream>
#include <vector>
#include <Shlwapi.h> 
#include <zip.h>
#include <unzip.h>
#include <zlib.h>

using namespace std;

#pragma comment(lib, "Shlwapi.lib")
#pragma comment(lib, "zlibstat.lib")

BOOL SaveToFile(char *pszFileName, BYTE *pData, DWORD dwDataSize)
{
	char szSaveName[MAX_PATH] = { 0 };
	lstrcpy(szSaveName, pszFileName);

	HANDLE hFile = CreateFile(szSaveName, GENERIC_READ | GENERIC_WRITE,
		FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, CREATE_ALWAYS,
		FILE_ATTRIBUTE_ARCHIVE, NULL);
	if (INVALID_HANDLE_VALUE == hFile)
	{
		return FALSE;
	}

	DWORD dwRet = 0;
	WriteFile(hFile, pData, dwDataSize, &dwRet, NULL);

	CloseHandle(hFile);

	return TRUE;
}

int main(int argc, char * argv[])
{
	char szBuffer[1024] = { 0 };

	strcpy(szBuffer, "test 123123");

	SaveToFile("d://test.txt", (BYTE *)szBuffer, sizeof(szBuffer));

	system("pause");
	return 0;
}

文件壓縮

compress 是 zlib 庫(kù)提供的用于數(shù)據(jù)壓縮的函數(shù),通過(guò)該函數(shù)可以將數(shù)據(jù)進(jìn)行壓縮。下面是一個(gè)示例,演示了如何使用 zlib 庫(kù)進(jìn)行文件壓縮。

它的原型如下:

int compress(Bytef* dest, uLongf* destLen, const Bytef* source, uLong sourceLen);
  • dest:指向存放壓縮后數(shù)據(jù)的緩沖區(qū)的指針。
  • destLen:傳入時(shí)為壓縮緩沖區(qū)的大小,傳出時(shí)為實(shí)際壓縮后數(shù)據(jù)的大小。
  • source:指向待壓縮數(shù)據(jù)的緩沖區(qū)的指針。
  • sourceLen:待壓縮數(shù)據(jù)的大小。

compress 函數(shù)的作用是將 source 指向的數(shù)據(jù)進(jìn)行壓縮,并將結(jié)果存放在 dest 指向的緩沖區(qū)中。destLen 傳入時(shí)應(yīng)該是 dest 緩沖區(qū)的大小,函數(shù)執(zhí)行后,destLen 會(huì)更新為實(shí)際壓縮后數(shù)據(jù)的大小。

函數(shù)返回值為壓縮的執(zhí)行狀態(tài),可能的返回值包括:

  • Z_OK:壓縮成功。
  • Z_MEM_ERROR:內(nèi)存分配失敗。
  • Z_BUF_ERROR:壓縮輸出緩沖區(qū)不足。

這個(gè)函數(shù)實(shí)際上是使用 DEFLATE 算法進(jìn)行壓縮,DEFLATE 是一種通用的壓縮算法,也是 zlib 庫(kù)的核心算法之一。壓縮后的數(shù)據(jù)可以使用 uncompress 函數(shù)進(jìn)行解壓縮。

總體而言,compress 函數(shù)提供了一種簡(jiǎn)單的方式,可以在應(yīng)用程序中對(duì)數(shù)據(jù)進(jìn)行壓縮,適用于需要減小數(shù)據(jù)體積的場(chǎng)景,比如網(wǎng)絡(luò)傳輸或數(shù)據(jù)存儲(chǔ)。

// 單個(gè)文件限制大小為 100M 
#define MAX_SRC_FILE_SIZE (100*1024*1024)

/**
 * @brief 壓縮指定文件的數(shù)據(jù)
 *
 * @param pszCompressFileName 待壓縮文件的路徑
 * @param ppCompressData 保存壓縮后數(shù)據(jù)的指針
 * @param pdwCompressDataSize 傳入時(shí)為壓縮緩沖區(qū)的大小,傳出時(shí)為實(shí)際壓縮后數(shù)據(jù)的大小
 * @return 壓縮是否成功,成功返回 TRUE,否則返回 FALSE
 */
BOOL CompressData(char *pszCompressFileName, BYTE **ppCompressData, DWORD *pdwCompressDataSize)
{
  HANDLE hFile = CreateFile(pszCompressFileName, GENERIC_READ,
    FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
    FILE_ATTRIBUTE_ARCHIVE, NULL);

  // 檢查文件句柄是否有效
  if (INVALID_HANDLE_VALUE == hFile)
  {
    return FALSE;
  }

  // 獲取文件大小
  DWORD dwFileSize = GetFileSize(hFile, NULL);

  // 檢查文件大小是否超過(guò)限制
  if (MAX_SRC_FILE_SIZE < dwFileSize)
  {
    CloseHandle(hFile);
    return FALSE;
  }

  DWORD dwDestDataSize = dwFileSize;

  // 分配源數(shù)據(jù)和目標(biāo)數(shù)據(jù)的內(nèi)存
  BYTE *pSrcData = new BYTE[dwFileSize];
  if (NULL == pSrcData)
  {
    CloseHandle(hFile);
    return FALSE;
  }

  BYTE *pDestData = new BYTE[dwDestDataSize];
  if (NULL == pDestData)
  {
    delete[] pSrcData;
    CloseHandle(hFile);
    return FALSE;
  }

  DWORD dwRet = 0;

  // 讀取源數(shù)據(jù)
  ReadFile(hFile, pSrcData, dwFileSize, &dwRet, NULL);

  // 檢查讀取是否成功
  if ((0 >= dwRet) || (dwRet != dwFileSize))
  {
    delete[] pDestData;
    delete[] pSrcData;
    CloseHandle(hFile);
    return FALSE;
  }

  int iRet = 0;

  // 壓縮數(shù)據(jù)
  do
  {
    iRet = compress(pDestData, &dwDestDataSize, pSrcData, dwFileSize);

    // 壓縮成功,退出循環(huán)
    if (0 == iRet)
    {
      break;
    }
    // 輸出緩沖區(qū)不足,增加緩沖區(qū)大小并重試
    else if (-5 == iRet)
    {
      delete[] pDestData;
      pDestData = NULL;
      dwDestDataSize = dwDestDataSize + (100 * 1024);
      pDestData = new BYTE[dwDestDataSize];

      // 分配新的目標(biāo)數(shù)據(jù)內(nèi)存
      if (NULL == pDestData)
      {
        delete[] pSrcData;
        CloseHandle(hFile);
        return FALSE;
      }
    }
    // 壓縮失敗,釋放內(nèi)存并返回失敗
    else
    {
      delete[] pDestData;
      pDestData = NULL;
      delete[] pSrcData;
      pSrcData = NULL;
      CloseHandle(hFile);
      return FALSE;
    }
  } while (TRUE);

  // 保存壓縮后數(shù)據(jù)的指針和實(shí)際大小
  *ppCompressData = pDestData;
  *pdwCompressDataSize = dwDestDataSize;

  // 釋放源數(shù)據(jù)內(nèi)存
  delete[] pSrcData;

  // 關(guān)閉文件句柄
  CloseHandle(hFile);

  // 返回壓縮成功
  return TRUE;
}

文件解壓縮

uncompress 函數(shù)是 zlib 庫(kù)提供的用于數(shù)據(jù)解壓縮的函數(shù),通過(guò)該函數(shù)可以將壓縮后的數(shù)據(jù)解壓縮還原。下面是一個(gè)示例,演示了如何使用 zlib 庫(kù)進(jìn)行文件解壓縮。

它的原型如下:

int uncompress(Bytef* dest, uLongf* destLen, const Bytef* source, uLong sourceLen);
  • dest:指向存放解壓縮后數(shù)據(jù)的緩沖區(qū)的指針。
  • destLen:傳入時(shí)為解壓縮緩沖區(qū)的大小,傳出時(shí)為實(shí)際解壓縮后數(shù)據(jù)的大小。
  • source:指向待解壓縮數(shù)據(jù)的緩沖區(qū)的指針。
  • sourceLen:待解壓縮數(shù)據(jù)的大小。

uncompress 函數(shù)的作用是將 source 指向的數(shù)據(jù)進(jìn)行解壓縮,并將結(jié)果存放在 dest 指向的緩沖區(qū)中。destLen 傳入時(shí)應(yīng)該是 dest 緩沖區(qū)的大小,函數(shù)執(zhí)行后,destLen 會(huì)更新為實(shí)際解壓縮后數(shù)據(jù)的大小。

函數(shù)返回值為解壓縮的執(zhí)行狀態(tài),可能的返回值包括:

  • Z_OK:解壓縮成功。
  • Z_MEM_ERROR:內(nèi)存分配失敗。
  • Z_BUF_ERROR:解壓縮輸出緩沖區(qū)不足。
  • Z_DATA_ERROR:輸入數(shù)據(jù)錯(cuò)誤或損壞。

uncompress 函數(shù)實(shí)際上是使用 DEFLATE 算法進(jìn)行解壓縮,與 compress 函數(shù)相對(duì)應(yīng)。這兩個(gè)函數(shù)共同構(gòu)成了 zlib 庫(kù)中的基本數(shù)據(jù)壓縮和解壓縮功能。

在實(shí)際應(yīng)用中,可以使用這兩個(gè)函數(shù)來(lái)處理需要壓縮和解壓縮的數(shù)據(jù),例如在網(wǎng)絡(luò)通信中減小數(shù)據(jù)傳輸量或在存儲(chǔ)數(shù)據(jù)時(shí)減小占用空間。

/**
 * @brief 解壓指定文件的數(shù)據(jù)
 *
 * @param pszUncompressFileName 待解壓文件的路徑
 * @param ppUncompressData 保存解壓后數(shù)據(jù)的指針
 * @param pdwUncompressDataSize 傳入時(shí)為解壓緩沖區(qū)的大小,傳出時(shí)為實(shí)際解壓后數(shù)據(jù)的大小
 * @return 解壓是否成功,成功返回 TRUE,否則返回 FALSE
 */
BOOL UncompressData(char *pszUncompressFileName, BYTE **ppUncompressData, DWORD *pdwUncompressDataSize)
{
  HANDLE hFile = CreateFile(pszUncompressFileName, GENERIC_READ,
    FILE_SHARE_READ | FILE_SHARE_WRITE, NULL, OPEN_EXISTING,
    FILE_ATTRIBUTE_ARCHIVE, NULL);

  // 檢查文件句柄是否有效
  if (INVALID_HANDLE_VALUE == hFile)
  {
    return FALSE;
  }

  // 獲取文件大小
  DWORD dwFileSize = GetFileSize(hFile, NULL);

  // 設(shè)置目標(biāo)數(shù)據(jù)緩沖區(qū)大小
  DWORD dwDestDataSize = MAX_SRC_FILE_SIZE;

  // 分配源數(shù)據(jù)和目標(biāo)數(shù)據(jù)的內(nèi)存
  BYTE *pSrcData = new BYTE[dwFileSize];
  if (NULL == pSrcData)
  {
    CloseHandle(hFile);
    return FALSE;
  }

  BYTE *pDestData = new BYTE[dwDestDataSize];
  if (NULL == pDestData)
  {
    delete[] pSrcData;
    CloseHandle(hFile);
    return FALSE;
  }

  DWORD dwRet = 0;

  // 讀取源數(shù)據(jù)
  ReadFile(hFile, pSrcData, dwFileSize, &dwRet, NULL);

  // 檢查讀取是否成功
  if ((0 >= dwRet) || (dwRet != dwFileSize))
  {
    delete[] pDestData;
    delete[] pSrcData;
    CloseHandle(hFile);
    return FALSE;
  }

  int iRet = 0;

  // 解壓縮數(shù)據(jù)
  do
  {
    iRet = uncompress(pDestData, &dwDestDataSize, pSrcData, dwFileSize);

    // 解壓縮成功,退出循環(huán)
    if (0 == iRet)
    {
      break;
    }
    // 輸出緩沖區(qū)不足,增加緩沖區(qū)大小并重試
    else if (-5 == iRet)
    {
      delete[] pDestData;
      pDestData = NULL;
      dwDestDataSize = dwDestDataSize + (100 * 1024);
      pDestData = new BYTE[dwDestDataSize];

      // 分配新的目標(biāo)數(shù)據(jù)內(nèi)存
      if (NULL == pDestData)
      {
        delete[] pSrcData;
        CloseHandle(hFile);
        return FALSE;
      }
    }
    // 解壓縮失敗,釋放內(nèi)存并返回失敗
    else
    {
      delete[] pDestData;
      pDestData = NULL;
      delete[] pSrcData;
      pSrcData = NULL;
      CloseHandle(hFile);
      return FALSE;
    }
  } while (TRUE);

  // 保存解壓后數(shù)據(jù)的指針和實(shí)際大小
  *ppUncompressData = pDestData;
  *pdwUncompressDataSize = dwDestDataSize;

  // 釋放源數(shù)據(jù)內(nèi)存
  delete[] pSrcData;

  // 關(guān)閉文件句柄
  CloseHandle(hFile);

  // 返回解壓成功
  return TRUE;
}

演示示例

下面是一個(gè)包含文件壓縮和解壓縮的完整示例,展示了如何將文件進(jìn)行壓縮保存,然后解壓還原。

調(diào)用CompressData壓縮文件,返回結(jié)果pCompressData存放文件內(nèi)存字節(jié),結(jié)果dwCompressDataSize存放長(zhǎng)度,并調(diào)用SaveToFile保存到本地。

int main(int argc, char* argv[])
{
  BOOL bRet = FALSE;
  BYTE *pCompressData = NULL;
  DWORD dwCompressDataSize = 0;

  // 壓縮文件
  bRet = CompressData("d:\\test.exe", &pCompressData, &dwCompressDataSize);
  if (TRUE == bRet)
  {
    std::cout << "已壓縮" << std::endl;
  }

  // 保存壓縮數(shù)據(jù)為文件
  bRet = SaveToFile("d:\\text.zlib", pCompressData, dwCompressDataSize);
  if (TRUE == bRet)
  {
    std::cout << "已保存到文件" << std::endl;
  }

  // 釋放內(nèi)存
  delete[]pCompressData;
  pCompressData = NULL;

  system("pause");
  return 0;
}

調(diào)用UncompressData解壓縮文件,返回結(jié)果pUncompressData存放文件內(nèi)存字節(jié),結(jié)果dwUncompressDataSize存放長(zhǎng)度,并調(diào)用SaveToFile保存到本地。

int main(int argc, char* argv[])
{
  BOOL bRet = FALSE;
  BYTE *pUncompressData = NULL;
  DWORD dwUncompressDataSize = 0;

  // 解壓文件
  bRet = UncompressData("d:\\test.zlib", &pUncompressData, &dwUncompressDataSize);
  if (TRUE == bRet)
  {
    std::cout << "已解壓" << std::endl;
  }

  // 保存解壓數(shù)據(jù)為文件
  bRet = SaveToFile("d:\\test.exe", pUncompressData, dwUncompressDataSize);
  if (TRUE == bRet)
  {
    std::cout << "已保存到文件" << std::endl;
  }

  // 釋放內(nèi)存
  delete[]pUncompressData;
  pUncompressData = NULL;

  system("pause");
  return 0;
}

編譯時(shí)可能會(huì)提示無(wú)法生成SAFESEH影響的報(bào)錯(cuò)信息,如下圖所示;

此時(shí)打開(kāi)項(xiàng)目屬性頁(yè),找到鏈接器,高級(jí)選項(xiàng)卡,將映像安全處理改為否即可,如下圖所示;

結(jié)論

通過(guò)使用 zlib 庫(kù),我們可以方便地在應(yīng)用程序中實(shí)現(xiàn)數(shù)據(jù)的壓縮和解壓縮功能。這對(duì)于需要減小數(shù)據(jù)傳輸量或在存儲(chǔ)數(shù)據(jù)時(shí)減小占用空間的場(chǎng)景非常有用。在實(shí)際應(yīng)用中,可以根據(jù)需要調(diào)整緩沖區(qū)大小和處理流程,以適應(yīng)不同的數(shù)據(jù)處理需求。

以上就是C/C++使用Zlib實(shí)現(xiàn)文件的壓縮與解壓的詳細(xì)內(nèi)容,更多關(guān)于C++ Zlib文件壓縮與解壓的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • 淺析C++的引用與const指針與各種傳遞方式

    淺析C++的引用與const指針與各種傳遞方式

    這篇文章主要介紹了淺析C++的引用與const指針與各種傳遞方式的相關(guān)資料,需要的朋友可以參考下
    2017-08-08
  • 讓我們一起來(lái)對(duì)C語(yǔ)言指針再分析

    讓我們一起來(lái)對(duì)C語(yǔ)言指針再分析

    這篇文章主要為大家詳細(xì)介紹C語(yǔ)言的指針,本文進(jìn)行了深度解析,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-01-01
  • C語(yǔ)言中判斷int,long型等變量是否賦值的方法詳解

    C語(yǔ)言中判斷int,long型等變量是否賦值的方法詳解

    聲明了int ,long型等局部變量,在利用一些方法給這些變量賦值之后,想判斷這些變量是不是真的被賦初值了,怎么辦
    2013-07-07
  • 基于Qt實(shí)現(xiàn)電子木魚(yú)小游戲

    基于Qt實(shí)現(xiàn)電子木魚(yú)小游戲

    今年最火爆的解壓小游戲電子木魚(yú),現(xiàn)在許多軟件都上架了這個(gè)小程序。我在網(wǎng)上看了一下基本上都是用py和Java寫(xiě)的,所以我用QT重新寫(xiě)了一下,作為小白練手項(xiàng)目非常適合,快跟隨小編一起學(xué)習(xí)一下吧
    2023-01-01
  • C語(yǔ)言函數(shù)調(diào)用堆棧詳情分析

    C語(yǔ)言函數(shù)調(diào)用堆棧詳情分析

    這篇文章主要介紹了C語(yǔ)言函數(shù)調(diào)用堆棧詳情分析,文章圍繞主題展開(kāi)詳細(xì)的內(nèi)容介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下
    2022-07-07
  • 距離詳解Linux下的UDP方式通訊

    距離詳解Linux下的UDP方式通訊

    這篇文章主要介紹了距離詳解Linux下的UDP方式通訊,是深入Linux系統(tǒng)編程中的基礎(chǔ),需要的朋友可以參考下
    2015-10-10
  • Qt為exe添加ico圖片的簡(jiǎn)單實(shí)現(xiàn)步驟

    Qt為exe添加ico圖片的簡(jiǎn)單實(shí)現(xiàn)步驟

    這篇文章主要給大家介紹了關(guān)于Qt為exe添加ico圖片的簡(jiǎn)單實(shí)現(xiàn)步驟,通過(guò)文中介紹的方法可以幫助大家實(shí)現(xiàn)這個(gè)自定義exe圖標(biāo)的效果,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下
    2022-07-07
  • C++菱形繼承和虛繼承的實(shí)現(xiàn)

    C++菱形繼承和虛繼承的實(shí)現(xiàn)

    本文主要介紹了C++菱形繼承和虛繼承的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-06-06
  • VScode配置C語(yǔ)言環(huán)境完整版(親測(cè)可用)

    VScode配置C語(yǔ)言環(huán)境完整版(親測(cè)可用)

    這篇文章主要介紹了VScode配置C語(yǔ)言環(huán)境完整版,本文通過(guò)圖文并茂的形式給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2021-08-08
  • C語(yǔ)言之整數(shù)劃分問(wèn)題(遞歸法)實(shí)例代碼

    C語(yǔ)言之整數(shù)劃分問(wèn)題(遞歸法)實(shí)例代碼

    這篇文章主要介紹了C語(yǔ)言之整數(shù)劃分問(wèn)題(遞歸法)實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下
    2017-02-02

最新評(píng)論