最近因为项目的需要,要对zip压缩文件进行批量解压。在网上查阅了相关的资料后,最终使用zlib开源库实现了该功能。本文将对zlib开源库进行简单介绍,并给出一个使用zlib开源库对zip压缩文件进行解压的示例程序。
1.zlib开源库
zlib是应用最广泛的压缩与解压缩zip文件的免费开源库,提供了数据压缩与解压缩的函式库。
zlib中最关键的函数有以下两个:
(1)int compress(Bytef *dest, uLongf *destLen, const Bytef *source, uLong sourceLen);
(2)int uncompress(Bytef *dest, uLongf *destLen, const Bytef *source, uLong sourceLen);
其中,函数compress()用于将源缓冲区数据压缩到目的缓冲区,函数uncompress()用于将源缓冲区数据解压到目的缓冲区。
由此可见,zlib只是一个针对gzip以及deflate算法的库,用于将一段内存压缩/解压之后放到另一段内存上,这离压缩/解压文件甚至文件夹的目标还很远。但是,它提供了一个叫做minizip的例子给出了操作zip文件的方法。
2.minizip简介
minizip是zlib的上层库,它封装了与zip文件相关的操作。
minizip中与解压缩相关的API有以下几个:
(1)unzFile unzOpen(const char *path);
(2)int unzClose(unzFile file);
(3)int unzGetGlobalInfo(unzFile file, unz_global_info *pglobal_info);
(4)int unzGoToNextFile(unzFile file);
(5)int unzGetCurrentFileInfo(unzFile file, unz_file_info *pfile_info, char *szFileName, uLong fileNameBufferSize,
void *extraField, uLong extraFieldBufferSize, char *szComment, uLong commentBufferSize);
(6)int unzOpenCurrentFile(unzFile file);
(7)int unzCloseCurrentFile(unzFile file);
(8)int unzReadCurrentFile(unzFile file, voidp buf, unsigned len);
有了以上这些API,我们就可以对zip文件进行解压缩了。一个完整的解压过程应该包含以下这些步骤:
(1)调用unzOpen()函数打开一个zip压缩文件,其参数是zip压缩文件的路径。
(2)调用unzGetGlobalInfo()函数来获取zip压缩文件的一些信息(如内部文件个数等),这些信息会保存在传入参数pglobal_info中。
(3)然后开始遍历zip文件中的内部文件,初始时会自动定位到第一个内部文件,处理完一个内部文件后可以使用unzGoToNextFile()函数来跳转到下一个内部文件。
(4)对于每个内部文件来说,可以先调用unzGetCurrentFileInfo()函数来获取该内部文件信息(如文件的路径、文件大小等),这些信息会保存在传入参数pfile_info中。
(5)调用unzOpenCurrentFile()函数打开该内部文件。
(6)调用unzReadCurrentFile()函数读取该内部文件内容。
(7)该内部文件读取完毕之后,调用unzCloseCurrentFile()函数对内部文件进行关闭。
(8)zip文件中的所有内部文件遍历完成之后,调用unzClose()函数关闭打开的zip压缩文件。
3.示例程序
了解了以上的内容之后,我们就可以编写程序使用zlib以及minizip对zip压缩文件进行解压缩了。
3.1加载相关的头文件及库文件
在使用zlib以及minizip之前,我们需要加载相关的头文件及库文件到工程中。需要加载的头文件有zlib.h、unzip.h、zip.h。需要加载的库文件有zlib.lib、minizip.lib。需要添加的动态链接库zlib1.dll。这些文件都可以从网上下载得到。
#include "zlib/zlib.h"
#include "zlib/unzip.h"
#include "zlib/zip.h"
#pragma comment(lib, "zlib.lib")
#pragma comment(lib, "minizip.lib")
3.2配置工程
因为zlib以及minizip是用C语言编写的,在VC6.0中使用时,需要对工程进行如下配置,否则会出现编译链接通不过的问题。
(1)在“工程”、“设置”中选择“连接”标签页,在“分类”中选择输入,在“忽略库”中加入MSVCRT。
(2)在“工程”、“设置”中选择“C/C++”标签页,在“分类”中选择Code Generation,在“Use run-time library”中选择“Debug Multithreaded DLL”。
(3)在“工程”、“设置”中选择“C/C++”标签页,在“分类”中选择常规,在“预处理程序定义”中加入_AFXDLL。
3.3示例程序
如下的示例程序演示了如何调用minizip中的API对zip文件进行解压。
/*
* 函数功能 : 解压zip文件
* 备 注 : 参数strFilePath表示zip压缩文件的路径
* 参数strTempPath表示要解压到的文件目录
* 作 者 : 博客园 依旧淡然(http://www.cnblogs.com/menlsh/)
*/
void CZlibDemoDlg::UnzipFile(CString strFilePath, CString strTempPath)
{
int nReturnValue; //打开zip文件
unzFile unzfile = unzOpen(strFilePath);
if(unzfile == NULL)
{
MessageBox("打开zip文件失败!", "提示", MB_OK|MB_ICONWARNING);
return;
} //获取zip文件的信息
unz_global_info* pGlobalInfo = new unz_global_info;
nReturnValue = unzGetGlobalInfo(unzfile, pGlobalInfo);
if(nReturnValue != UNZ_OK)
{
MessageBox("获取zip文件信息失败!", "提示", MB_OK|MB_ICONWARNING);
return;
} //解析zip文件
unz_file_info* pFileInfo = new unz_file_info;
char szZipFName[MAX_PATH]; //存放从zip中解析出来的内部文件名
for(int i=; i<pGlobalInfo->number_entry; i++)
{
//解析得到zip中的文件信息
nReturnValue = unzGetCurrentFileInfo(unzfile, pFileInfo, szZipFName, MAX_PATH,
NULL, , NULL, );
if(nReturnValue != UNZ_OK)
{
MessageBox("解析zip文件信息失败!", "提示", MB_OK|MB_ICONWARNING);
return;
} //判断是文件夹还是文件
switch(pFileInfo->external_fa)
{
case FILE_ATTRIBUTE_DIRECTORY: //文件夹
{
CString strDiskPath = strTempPath + _T("//") + szZipFName;
CreateDirectory(strDiskPath, NULL);
}
break;
default: //文件
{
//创建文件
CString strDiskFile = strTempPath + _T("//") + szZipFName;
HANDLE hFile = CreateFile(strDiskFile, GENERIC_WRITE,
, NULL, OPEN_ALWAYS, FILE_FLAG_WRITE_THROUGH, NULL);
if(hFile == INVALID_HANDLE_VALUE)
{
MessageBox("创建文件失败!", "提示", MB_OK|MB_ICONWARNING);
return;
} //打开文件
nReturnValue = unzOpenCurrentFile(unzfile);
if(nReturnValue != UNZ_OK)
{
MessageBox("打开文件失败!", "提示", MB_OK|MB_ICONWARNING);
CloseHandle(hFile);
return;
} //读取文件
const int BUFFER_SIZE = ;
char szReadBuffer[BUFFER_SIZE];
while(TRUE)
{
memset(szReadBuffer, , BUFFER_SIZE);
int nReadFileSize = unzReadCurrentFile(unzfile, szReadBuffer, BUFFER_SIZE);
if(nReadFileSize < ) //读取文件失败
{
MessageBox("读取文件失败!", "提示", MB_OK|MB_ICONWARNING);
unzCloseCurrentFile(unzfile);
CloseHandle(hFile);
return;
}
else if(nReadFileSize == ) //读取文件完毕
{
unzCloseCurrentFile(unzfile);
CloseHandle(hFile);
break;
}
else //写入读取的内容
{
DWORD dWrite = ;
BOOL bWriteSuccessed = WriteFile(hFile, szReadBuffer, BUFFER_SIZE, &dWrite, NULL);
if(!bWriteSuccessed)
{
MessageBox("读取文件失败!", "提示", MB_OK|MB_ICONWARNING);
unzCloseCurrentFile(unzfile);
CloseHandle(hFile);
return;
}
}
}
}
break;
}
unzGoToNextFile(unzfile);
} //关闭
if(unzfile)
{
unzClose(unzfile);
}
}
3.4运行结果
调用上述的UnzipFile()方法对某个zip文件进行解压,如图1所示。
图1 解压zip文件
解压后,可以看到文件夹123中的内容如图2所示。
图2 解压后的文件