[Qt]简单使用 Zlib 库压缩/解压数据

一直想解码酷狗的krc格式文件,但是苦于没有找到资料,最近在吾爱破解发现了一篇文章讲解码krc文件的(地址),我就打算自己试试。文中解码使用了 zlib 库,所以便看看如何使用Zlib库,也算是积累一点知识吧。
在写这些代码的时候,遇到了一些问题,就包括上一篇讲的Qt断点无法断下,还有QByteArray的对象转char *时会截断'\0'(我最想吐槽这一点,花了我很长时间在解压函数上)。 

2015.6.29 Note
ZLibGetDecodeLength被用在ZlibUncompress函数中仍然有空间不足的危险, Zlib建议设置一个固定的缓冲区,且缓冲区大小与解压速度相关联.

/************************************************

 * [Qt]简单使用 Zlib 库压缩/解压数据

 * DownLoad: http://be18e3.l5.yunpan.cn/lk/cQW55yPJ8b78r (2c5b)

 * By sunnysab   2015.6.18 - 6.28

************************************************/



#include <QCoreApplication>

#include <QByteArray>



#include "zlib.h"



/*

 * 函  数: ZlibGetDecodeLength

 * 功  能: 通过要压缩的数据大小,得到压缩后数据的最大大小

 *        以此设置缓冲区

 * 参  数: uLong  InRawLength  压缩前数据大小

 *

 */

int  ZlibGetDecodeLength(uLong  InRawLength)

{

    return  compressBound(InRawLength);

}





/*

 * 函  数: ZlibGetDecodeLength

 * 功  能: 通过要压缩的数据大小,得到压缩后数据的最大大小

 *        以此设置缓冲区

 * 参  数: QByteArray  InRawData  压缩前数据

 *

 */

int  ZlibGetDecodeLength(QByteArray  InRawData)

{

    // 注:

    // ((QByteArray)"Hello").size() == 5

    // 因为当字符串为 QByteArray , '\0' 被忽略



    return  compressBound(InRawData.length());

}





/*

 * 函  数: ZlibCompress

 * 功  能: 执行压缩操作

 * 参  数: QByteArray  &OutEncodeData  压缩后数据

 *        QByteArray  InRawData       要压缩的数据

          int       * nErrorCode      错误代码

 * 说  明: 当 nErrorCode 不为 NULL 时, 若 compress 出错后, 其数据改为

 *          compress 返回的错误代码. 在 zlib.h 中可以找到.

 *        例如: -5  缓冲区长度不够

 * 返回值: 类型 long

 *        返回压缩后的长度, -1 表示压缩失败, 请检查 *nErrorCode .

 */

long  ZlibCompress(QByteArray  &OutEncodeData, QByteArray  InRawData,

                  int  * nErrorCode = NULL)

{

    Bytef  *EncodeData = NULL;  // 压缩后的数据暂存于此

    int     nFuncRet = Z_ERRNO; // 随便找个错误代码先填着

    long    nOutLength = 0;



    // 申请最大缓冲区

    nOutLength = ZlibGetDecodeLength(InRawData.length());

    EncodeData = new Bytef[nOutLength];



    if (NULL != EncodeData)

    {

        // 调用 compress 函数前, nOutLength 要置为缓冲区大小

        nFuncRet = compress(EncodeData, (uLongf *)&nOutLength, (Bytef *)InRawData.data(),

                            InRawData.length());



        if (Z_OK == nFuncRet)

        {

            // 自己设置好长度为好, 省的 '\0' 被截断了

            OutEncodeData.append((const char *)EncodeData, nOutLength);

        }

        else

        {

            nOutLength = -1;



        }

    }



    delete  EncodeData;



    // 输出错误代码

    if (NULL != nErrorCode)

    {

        *nErrorCode = nFuncRet;

    }

    return  nOutLength;

}





/*

 * 函  数: ZlibUnCompress

 * 功  能: 执行解压操作

 * 参  数: QByteArray  &OutDecodeData  解压后数据

 *        QByteArray  InEncodeData    要解压的数据

          int       * nErrorCode      错误代码

 * 说  明: 当 nErrorCode 不为 NULL 时, 若 uncompress 出错后, 其数据改为

 *          uncompress 返回的错误代码. 在 zlib.h 中可以找到.

 *        例如: -5  缓冲区长度不够

 * 返回值: 类型 long

 *        返回解压后的长度, -1 表示解压失败, 请检查 *nErrorCode .

 */

long  ZlibUncompress(QByteArray  &OutDecodeData, QByteArray InEncodeData,

                     int   * nErrorCode = NULL)

{

    Bytef  *DecodeData = NULL; // 解压后的数据缓冲区

    Bytef  *EncodeData_Buffer = NULL;  // 要解压的数据缓冲区, 后面用memcpy放到这

    int     nFuncRet = Z_ERRNO;

    unsigned long  nOutLength = 0;



    // 说明:

    // 如果压缩后大小比之前小, 缓冲区大小应该设置多少?

    // 暂且把 ZlibGetDecodeLength 拿来用用, 确保缓冲区够了.

    nOutLength = ZlibGetDecodeLength(InEncodeData.length());



    // 为缓冲区申请内存

    DecodeData        =  new Bytef[nOutLength];

    EncodeData_Buffer =  new Bytef[nOutLength];



    if (NULL != DecodeData && NULL != EncodeData_Buffer)

    {

        // 由于 QByteArray 在 data() 函数中会因为'\0'截断

        // (先鄙视一下,那还叫什么字节操作)

        // 文档又说,constData() 效率高

        // 所以用 constData() 获取地址然后 memcpy 自己复制

        memcpy(EncodeData_Buffer, InEncodeData.constData(), InEncodeData.size());



        nFuncRet = uncompress(DecodeData, &nOutLength, (Bytef *)EncodeData_Buffer,

                              InEncodeData.length());



        if (Z_OK == nFuncRet)

        {

            OutDecodeData.append((const char *)DecodeData, nOutLength);

        }

        else

        {

            nOutLength = -1;

        }

    }

    else

    {

        nOutLength = -1;

    }



    // 清理 new 产生的内存

    if (NULL != DecodeData)

    {

        delete  DecodeData;

    }

    if (NULL != EncodeData_Buffer)

    {

        delete  EncodeData_Buffer;

    }

    // 输出错误代码

    if (NULL != nErrorCode)

    {

        *nErrorCode = nFuncRet;

    }



    return  nOutLength;

}



int  main(int argc, char *argv[])

{

    QCoreApplication AppInstance(argc, argv);

    QByteArray  strEncode;

    QByteArray  strDecode;

    int nFuncRet = 0;



    nFuncRet = ZlibCompress(strEncode, (QByteArray)"Hello World");

    printf("retCompress = %d\n", nFuncRet);



    nFuncRet = ZlibUncompress(strDecode, strEncode);

    printf("retUnCompress = %d\n", nFuncRet);



    printf("data = %s\n", strDecode.data());



    return AppInstance.exec();

}
  • 5
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 6
    评论
在Windows下使用Qt进行文件夹压缩解压缩,可以使用zlib进行实现。以下是实现的基本步骤: 1. 下载zlib并将其添加到Qt项目中。 2. 使用zlib的函数进行文件夹压缩,代码示例: ```cpp #include <zlib.h> // 压缩文件夹 bool compressFolder(QString folderPath, QString zipFilePath) { QDir dir(folderPath); if (!dir.exists()) { qDebug() << "Folder not exists!"; return false; } QFile zipFile(zipFilePath); if (!zipFile.open(QIODevice::WriteOnly)) { qDebug() << "Failed to create zip file!"; return false; } // 初始化zlib stream z_stream stream; stream.zalloc = Z_NULL; stream.zfree = Z_NULL; stream.opaque = Z_NULL; deflateInit(&stream, Z_DEFAULT_COMPRESSION); // 遍历文件夹下所有文件进行压缩 QList<QFileInfo> fileInfoList = dir.entryInfoList(QDir::Files | QDir::Dirs | QDir::NoDotAndDotDot); foreach (QFileInfo fileInfo, fileInfoList) { if (fileInfo.isDir()) { QString subFolderPath = fileInfo.filePath(); QString subZipFilePath = zipFilePath + "/" + fileInfo.fileName() + ".zip"; compressFolder(subFolderPath, subZipFilePath); } else { QFile file(fileInfo.filePath()); if (!file.open(QIODevice::ReadOnly)) { qDebug() << "Failed to open file: " << fileInfo.filePath(); continue; } QByteArray data = file.readAll(); QByteArray compressedData(data.size(), Qt::Uninitialized); stream.avail_in = data.size(); stream.next_in = (Bytef *)data.data(); stream.avail_out = compressedData.size(); stream.next_out = (Bytef *)compressedData.data(); deflate(&stream, Z_FINISH); compressedData.resize(stream.total_out); zipFile.write(compressedData); } } // 完成压缩 deflateEnd(&stream); zipFile.close(); return true; } ``` 3. 使用zlib的函数进行文件夹解压缩,代码示例: ```cpp #include <zlib.h> // 解压文件夹 bool decompressFolder(QString zipFilePath, QString folderPath) { QFile zipFile(zipFilePath); if (!zipFile.open(QIODevice::ReadOnly)) { qDebug() << "Failed to open zip file!"; return false; } QDir dir(folderPath); if (!dir.exists()) { dir.mkpath("."); } // 初始化zlib stream z_stream stream; stream.zalloc = Z_NULL; stream.zfree = Z_NULL; stream.opaque = Z_NULL; inflateInit(&stream); // 逐个解压文件 QByteArray buffer(1024 * 1024, Qt::Uninitialized); while (!zipFile.atEnd()) { QByteArray compressedData = zipFile.read(buffer.size()); buffer.resize(compressedData.size() * 2); stream.avail_in = compressedData.size(); stream.next_in = (Bytef *)compressedData.data(); stream.avail_out = buffer.size(); stream.next_out = (Bytef *)buffer.data(); int result = inflate(&stream, Z_SYNC_FLUSH); if (result != Z_OK && result != Z_STREAM_END) { qDebug() << "Failed to decompress data!"; inflateEnd(&stream); return false; } buffer.resize(buffer.size() - stream.avail_out); QString fileName = QFileInfo(zipFilePath).baseName(); QFile file(folderPath + "/" + fileName); if (!file.open(QIODevice::WriteOnly | QIODevice::Append)) { qDebug() << "Failed to create file: " << folderPath + "/" + fileName; continue; } file.write(buffer); file.close(); } // 完成解压缩 inflateEnd(&stream); zipFile.close(); return true; } ``` 这样,就可以使用zlib进行文件夹压缩解压缩了。
评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值