我在Windows平台上用C ++编写程序。我想压缩存储在char[]
数组中的一些数据,并将其输出到文件中,稍后我将文件上传到unix服务器,我希望它可以由gzip -d
解压缩。 / p>
经过大量研究,我选择miniz。另外,我找到了gzip文件格式here。
以下是创建gzip文件的代码片段:(抱歉,我没有提供某些变量的定义;它们已在其他地方定义)
unsigned long zsize;
zpkg[0] = 0x1F;
zpkg[1] = 0x8B;
zpkg[2] = 8;
zpkg[3] = 0;
zpkg[4] = 0;
zpkg[5] = 0;
zpkg[6] = 0;
zpkg[7] = 0;
zpkg[8] = 0;
zpkg[9] = 0xFF;
compress2(zpkg + 10, &zsize, pkg, pkgSize, MZ_DEFAULT_LEVEL);
int footerStart = (int)zsize + 10;
mz_ulong crc = mz_crc32(MZ_CRC32_INIT, zpkg + 10, zsize);
zpkg[footerStart] = crc & 0xFF;
zpkg[footerStart + 1] = (crc >> 8) & 0xFF;
zpkg[footerStart + 2] = (crc >> 16) & 0xFF;
zpkg[footerStart + 3] = (crc >> 24) & 0xFF;
zpkg[footerStart + 4] = pkgSize & 0xFF;
zpkg[footerStart + 5] = (pkgSize >> 8) & 0xFF;
zpkg[footerStart + 6] = (pkgSize >> 16) & 0xFF;
zpkg[footerStart + 7] = (pkgSize >> 24) & 0xFF;
稍后只需将zpkg
数组输出到文件中。但这不起作用;当我用gzip解压缩时,错误消息是:
gzip: data stream error
gzip: test.gz: uncompress failed
有人可以指出我做错了吗?
感谢Mark Adler和Michael,我找到了一个有效的解决方案。
首先,正如Mark指出的那样,我应该让miniz返回一个原始的deflate数据流。这可以通过将-MZ_DEFAULT_WINDOW_BITS
(注意减号)传递给mz_deflateInit2()
作为第四个参数来完成。查看迷你源代码,compress2()
函数最终使用mz_deflateInit2()
调用MZ_DEFAULT_WINDOW_BITS
,这意味着添加zlib页眉和页脚。所以最简单的解决方法是在那里添加一个减号,这样我仍然可以使用compress2()
函数。 (这对我有用,因为我只在一个地方调用此功能)
其次,正如迈克尔指出的那样,应该在未压缩数据上计算CRC码。所以我这样修理:
mz_ulong crc = mz_crc32(MZ_CRC32_INIT, pkg, pkgSize);
完成上述两项更改后,gzip -d
不会再抱怨了。
答案 0 :(得分:1)
compress2()
生成一个zlib流,它使用zlib头和尾部来压缩压缩数据。对于您正在做的事情,您只需要将原始的deflate压缩流粘贴到手动生成的gzip标头和预告片中。
您可以:a)丢弃compress2()
输出的前两个和后四个字节以去除zlib标头和预告片,b)使用deflateInit2()
,deflate()
和deflateEnd()
而不是compress2()
并选择原始deflate格式,或c)使用相同的函数,而是选择gzip格式,并从deflate()
开始删除手动构建的gzip标头和预告片会为你做那件事。
我推荐c)。