QT QZipReader改进,以支持大于2G的zip文件

2024-04-18 20:52

本文主要是介绍QT QZipReader改进,以支持大于2G的zip文件,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

QZipReader对ZIP文件读取非常方便好用。即使在最新版的QT 6.6.1里,仍然存在一些问题:对于大于2G的zip文件不支持。

虽然有标准zlib可调用,但包装成一个易用且功能成熟的zip解压功能库,还是有很大的工作量,也需要有一定的经验。

于是,直接找到QT的QZipReader相关的源码文件,单独做成一个Compress工具包,方便BUG调试,其次,将来再整合进7z解压等功能。

把原来的两个头文件和一个Cpp文件,整合成两个文件:QZip.h和QZip.cpp。

把QZipReader改名为ZipReader,以避免与QT自带的头文件和DLL产生冲突。

代码工程结构如下图:

代码调试过程,就不截图了。主要修改了两个方法:

1、void ZipReaderPrivate::scanFiles()

把这些变量的类型由int纠正为正确的数据型,这就是不支持超过2G的zip文件原因。

        qint64 i = 0;
    uint start_of_directory = -1;
    ushort num_dir_entries = 0;

说明原来代码还是有一些基本规范质量问题,也不知道是有意的,还是无意的,或者是发现问题了,但没有人手去分析与修改。

2、QByteArray ZipReader::fileData(const QString &fileName)

问题跟前一个方法一样,把一些变量的类型纠正为正确的数据型。

修改后的源代码:

void ZipReaderPrivate::scanFiles()

void ZipReaderPrivate::scanFiles()
{if (!dirtyFileTree)return;if (! (device->isOpen() || device->open(QIODevice::ReadOnly))) {status = ZipReader::FileOpenError;return;}if ((device->openMode() & QIODevice::ReadOnly) == 0) { // only read the index from readable files.status = ZipReader::FileReadError;return;}dirtyFileTree = false;uchar tmp[4];device->read((char *)tmp, 4);if (readUInt(tmp) != 0x04034b50) {qWarning("QZip: not a zip file!");return;}// find EndOfDirectory headerqint64 i = 0;uint start_of_directory = -1;ushort num_dir_entries = 0;EndOfDirectory eod;while (true) {const qint64 pos = device->size() - qint64(sizeof(EndOfDirectory)) - i;if (pos < 0 || i > 65535) {debugx("Zip: EndOfDirectory not found。 ");return;}device->seek(pos);device->read((char *)&eod, sizeof(EndOfDirectory));if (readUInt(eod.signature) == 0x06054b50)break;++i;}// have the eodstart_of_directory = readUInt(eod.dir_start_offset);num_dir_entries = readUShort(eod.num_dir_entries);debugx("start_of_directory at %u, num_dir_entries=%d", start_of_directory, num_dir_entries);int comment_length = readUShort(eod.comment_length);if (comment_length != i)debugx("QZip: failed to parse zip file.");comment = device->read(qMin(comment_length, i));device->seek(start_of_directory);for (i = 0; i < num_dir_entries; ++i) {FileHeader header;auto read = device->read((char *) &header.h, sizeof(CentralFileHeader));if (read < (qint64)sizeof(CentralFileHeader)) {debugx("QZip: Failed to read complete header, index may be incomplete,read:%lld,num_dir_entries:%d",read,num_dir_entries);break;}if (readUInt(header.h.signature) != 0x02014b50) {debugx("QZip: invalid header signature, index may be incomplete");break;}int l = readUShort(header.h.file_name_length);header.file_name = device->read(l);if (header.file_name.size() != l) {debugx("QZip: Failed to read filename from zip index, index may be incomplete");break;}l = readUShort(header.h.extra_field_length);header.extra_field = device->read(l);if (header.extra_field.size() != l) {debugx("QZip: Failed to read extra field in zip file, skipping file, index may be incomplete");break;}l = readUShort(header.h.file_comment_length);header.file_comment = device->read(l);if (header.file_comment.size() != l) {debugx("QZip: Failed to read read file comment, index may be incomplete");break;}//  debugx("found file '%s'", header.file_name.data());fileHeaders.append(header);}
}

QByteArray ZipReader::fileData(const QString &fileName)

QByteArray ZipReader::fileData(const QString &fileName) const
{d->scanFiles();int i;for (i = 0; i < d->fileHeaders.size(); ++i) {if (QString::fromLocal8Bit(d->fileHeaders.at(i).file_name) == fileName)break;}if (i == d->fileHeaders.size())return QByteArray();FileHeader header = d->fileHeaders.at(i);ushort version_needed = readUShort(header.h.version_needed);if (version_needed > ZIP_VERSION) {debugx("QZip: .ZIP specification version %d implementationis needed to extract the data.", version_needed);return QByteArray();}ushort general_purpose_bits = readUShort(header.h.general_purpose_bits);uint compressed_size = readUInt(header.h.compressed_size);uint uncompressed_size = readUInt(header.h.uncompressed_size);uint start = readUInt(header.h.offset_local_header);//qDebug("uncompressing file %d: local header at %d", i, start);d->device->seek(start);LocalFileHeader lh;d->device->read((char *)&lh, sizeof(LocalFileHeader));uint skip = readUShort(lh.file_name_length) + readUShort(lh.extra_field_length);d->device->seek(d->device->pos() + skip);ushort compression_method = readUShort(lh.compression_method);//qDebug("file=%s: compressed_size=%d, uncompressed_size=%d", fileName.toLocal8Bit().data(), compressed_size, uncompressed_size);if ((general_purpose_bits & Encrypted) != 0) {debugx("QZip: Unsupported encryption method is needed to extract the data.");return QByteArray();}//qDebug("file at %lld", d->device->pos());QByteArray compressed = d->device->read(compressed_size);if (compression_method == CompressionMethodStored) {// no compressioncompressed.truncate(uncompressed_size);return compressed;} else if (compression_method == CompressionMethodDeflated) {// Deflate//qDebug("compressed=%d", compressed.size());compressed.truncate(compressed_size);QByteArray baunzip;ulong len = qMax(uncompressed_size,  1u);int res;do {baunzip.resize(len);res = inflate((uchar*)baunzip.data(), &len,(const uchar*)compressed.constData(), compressed_size);switch (res) {case Z_OK:if ((int)len != baunzip.size())baunzip.resize(len);break;case Z_MEM_ERROR:debugx("QZip: Z_MEM_ERROR: Not enough memory");break;case Z_BUF_ERROR:len *= 2;break;case Z_DATA_ERROR:debugx("QZip: Z_DATA_ERROR: Input data is corrupted");break;}} while (res == Z_BUF_ERROR);return baunzip;}debugx("Zip: Unsupported compression method %d is needed to extract the data.", compression_method);return QByteArray();
}

调用示例:(功能,直接读取zip包里的指定的图像文件的内容)

Mat readImage(QString filePath,bool isZip, int flags)
{if(isZip){auto index= filePath.indexOf(".zip");QString zipFile=filePath.left(index+4);QString imgName=filePath.right(filePath.length()-index-5);//      debugx2(zipFile<<"\n"<<imgName);ZipReader zipreader(zipFile);for(auto fileInfo : zipreader.fileInfoList()){if(fileInfo.isFile){if(fileInfo.filePath==imgName){debugx2(fileInfo.filePath);//注意编码问题
//                    QByteArray dt = fileInfo.filePath.toUtf8();
//                    QString strtemp = QString::fromLocal8Bit(dt);QByteArray array = zipreader.fileData(fileInfo.filePath);//   debugx2(array.size());//   imshow("image",image);zipreader.close();if(array.size()>0)return  imdecode(std::vector<char>(array.constData(),array.constData()+array.size()),flags);else{debugx2(zipFile<<" read fail: "<<imgName);return Mat();}}}}zipreader.close();return Mat();}Mat src = cv::imread(filePath.toLocal8Bit().data(),flags);return src;
}

插件运行效果:不用解压zip,就可以直接浏览zip包里的图片。

这篇关于QT QZipReader改进,以支持大于2G的zip文件的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/915820

相关文章

基于Qt Qml实现时间轴组件

《基于QtQml实现时间轴组件》时间轴组件是现代用户界面中常见的元素,用于按时间顺序展示事件,本文主要为大家详细介绍了如何使用Qml实现一个简单的时间轴组件,需要的可以参考下... 目录写在前面效果图组件概述实现细节1. 组件结构2. 属性定义3. 数据模型4. 事件项的添加和排序5. 事件项的渲染如何使用

基于Qt开发一个简单的OFD阅读器

《基于Qt开发一个简单的OFD阅读器》这篇文章主要为大家详细介绍了如何使用Qt框架开发一个功能强大且性能优异的OFD阅读器,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 目录摘要引言一、OFD文件格式解析二、文档结构解析三、页面渲染四、用户交互五、性能优化六、示例代码七、未来发展方向八、结论摘要

python与QT联合的详细步骤记录

《python与QT联合的详细步骤记录》:本文主要介绍python与QT联合的详细步骤,文章还展示了如何在Python中调用QT的.ui文件来实现GUI界面,并介绍了多窗口的应用,文中通过代码介绍... 目录一、文章简介二、安装pyqt5三、GUI页面设计四、python的使用python文件创建pytho

QT实现TCP客户端自动连接

《QT实现TCP客户端自动连接》这篇文章主要为大家详细介绍了QT中一个TCP客户端自动连接的测试模型,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录版本 1:没有取消按钮 测试效果测试代码版本 2:有取消按钮测试效果测试代码版本 1:没有取消按钮 测试效果缺陷:无法手动停

基于Qt实现系统主题感知功能

《基于Qt实现系统主题感知功能》在现代桌面应用程序开发中,系统主题感知是一项重要的功能,它使得应用程序能够根据用户的系统主题设置(如深色模式或浅色模式)自动调整其外观,Qt作为一个跨平台的C++图形用... 目录【正文开始】一、使用效果二、系统主题感知助手类(SystemThemeHelper)三、实现细节

Qt实现文件的压缩和解压缩操作

《Qt实现文件的压缩和解压缩操作》这篇文章主要为大家详细介绍了如何使用Qt库中的QZipReader和QZipWriter实现文件的压缩和解压缩功能,文中的示例代码简洁易懂,需要的可以参考一下... 目录一、实现方式二、具体步骤1、在.pro文件中添加模块gui-private2、通过QObject方式创建

Qt QWidget实现图片旋转动画

《QtQWidget实现图片旋转动画》这篇文章主要为大家详细介绍了如何使用了Qt和QWidget实现图片旋转动画效果,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 一、效果展示二、源码分享本例程通过QGraphicsView实现svg格式图片旋转。.hpjavascript

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

【QT】基础入门学习

文章目录 浅析Qt应用程序的主函数使用qDebug()函数常用快捷键Qt 编码风格信号槽连接模型实现方案 信号和槽的工作机制Qt对象树机制 浅析Qt应用程序的主函数 #include "mywindow.h"#include <QApplication>// 程序的入口int main(int argc, char *argv[]){// argc是命令行参数个数,argv是

Python QT实现A-star寻路算法

目录 1、界面使用方法 2、注意事项 3、补充说明 用Qt5搭建一个图形化测试寻路算法的测试环境。 1、界面使用方法 设定起点: 鼠标左键双击,设定红色的起点。左键双击设定起点,用红色标记。 设定终点: 鼠标右键双击,设定蓝色的终点。右键双击设定终点,用蓝色标记。 设置障碍点: 鼠标左键或者右键按着不放,拖动可以设置黑色的障碍点。按住左键或右键并拖动,设置一系列黑色障碍点