c - 如何将放气/膨胀 SetDictionary 与原始放气/膨胀一起使用?

标签 c zlib compression deflate

当我们使用字典时,我试图理解原始压缩的功能。我知道以下内容。 1. 当我们使用字典时,应用程序应该为 deflate() 和 inflate() 提供相同的字典。 2. 进行原始放气时,必须在任何放气调用之前或放气 block 完成后立即调用此函数,即在使用任何冲洗选项时消耗所有输入并交付所有输出之后 Z_BLOCKZ_PARTIAL_FLUSHZ_SYNC_FLUSHZ_FULL_FLUSH。 (来自 zlib 文档)。

但是下面的应用程序无法解压缩之前使用同一应用程序压缩的内容。压缩和解压缩成功,但输入和未压缩文件不匹配。

放气:

    do {
        ret = deflateSetDictionary(&strm, dictionary, sizeof(dictionary));
        if(ret != Z_OK) {
            fprintf(stderr, "Failed to set deflate dictionary\n");
            return Z_STREAM_ERROR;
        }
        strm.avail_in = fread(in, 1, CHUNK, source);
        if (ferror(source)) {
            (void)deflateEnd(&strm);
            return Z_ERRNO;
        }
        flush = feof(source) ? Z_FINISH : Z_FULL_FLUSH;
        strm.next_in = in;

        /* run deflate() on input until output buffer not full, finish
           compression if all of source has been read in */
        do {
            strm.avail_out = CHUNK;
            strm.next_out = out;
            ret = deflate(&strm, flush);    /* no bad return value */
            assert(ret != Z_STREAM_ERROR);  /* state not clobbered */
            have = CHUNK - strm.avail_out;
            if (fwrite(out, 1, have, dest) != have || ferror(dest)) {
                (void)deflateEnd(&strm);
                return Z_ERRNO;
            }
        } while (strm.avail_out == 0);
        assert(strm.avail_in == 0);     /* all input will be used */

        /* done when last data in file processed */
    } while (flush != Z_FINISH);
    assert(ret == Z_STREAM_END);     

膨胀:

    do {
        ret = inflateSetDictionary(&strm, dictionary, sizeof(dictionary));
        if(ret != Z_OK) {
            fprintf(stderr, "Failed to set inflate dictionary\n");
            return Z_STREAM_ERROR;
        }
        strm.avail_in = fread(in, 1, CHUNK, source);
        if (ferror(source)) {
            (void)inflateEnd(&strm);
            return Z_ERRNO;
        }
        if (strm.avail_in == 0)
            break;
        strm.next_in = in;


        /* run inflate() on input until output buffer not full */
        do {
            strm.avail_out = CHUNK;
            strm.next_out = out;
            ret = inflate(&strm, Z_FULL_FLUSH);
            assert(ret != Z_STREAM_ERROR);  /* state not clobbered */
            switch (ret) {
            case Z_NEED_DICT:
                ret = Z_DATA_ERROR;     /* and fall through */
            case Z_DATA_ERROR:
            case Z_MEM_ERROR:
                (void)inflateEnd(&strm);
                return ret;
            }
            have = CHUNK - strm.avail_out;
            if (fwrite(out, 1, have, dest) != have || ferror(dest)) {
                (void)inflateEnd(&strm);
                return Z_ERRNO;
            }
        } while (strm.avail_out == 0);

        /* done when inflate() says it's done */
    } while (ret != Z_STREAM_END);

最佳答案

放气时,您为每个 CHUNK 输入字节设置相同的字典。为什么?您应该在 deflateInit2() 之后使用一次 deflateSetDictionary()。从那时起,输入数据本身应该作为比您可能提供的字典更好的匹配字符串来源。

在膨胀方面,您必须知道压缩 block 的结束位置,以便您可以在与压缩时完全相同的位置执行 inflateSetDictionary()。这将需要某种标记、计数或搜索完整冲洗模式。

关于c - 如何将放气/膨胀 SetDictionary 与原始放气/膨胀一起使用?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30702486/

相关文章:

c++ - 使用 zlib 将字节数组解压缩为字节数组

node.js - zlib:zlib 绑定(bind)​​已关闭

javascript - 在生产环境中调试 Uglified javascript

ruby-on-rails - Rails回形针图像压缩与Page Speed产生的图像相比

c++ - 为什么 C 或 C++ 标准不明确将 char 定义为有符号或无符号?

python - 在 Google App Engine 上存储之前压缩数据

c - 循环中c中的随机数

javascript - asp.net 中 css 和 js 的 GZIP 压缩

C如何将指针传递给结构数组指针

c - 在 time.h 中找不到 timespec