LZW编码与解码的那点事

1. 前言

在前面一片文章TIFF文件结构详解里,我们在解析TIF文件中,提到使用了LZW算法。那么LZW算法是什么呢?就让我们揭开它的神秘面纱。

2. 什么是LZW

LZW算法又叫“串表压缩算法”就是通过建立一个字符串表,用较短的代码来表示较长的字符串来实现压缩。 LZW压缩算法是Unisys的专利,有效期到2003年,所以对它的使用已经没有限制了。字符串和编码的对应关系是在压缩过程中动态生成的,并且隐含在压缩数据中,解压的时候根据表来进行恢复,算是一种无损压缩。这里涉及到对应关系表,最开始会有一个单个字符的对应关系表,当然,一般都以ACCISS码表示,这里提到的关系映射表主要是多个字符的关系映射表。

3. LZW编码算法

3.1 概述

编码器从原字符串不断地读入新的字符,并试图将单个字符或字符串编码为记号 (Symbol)。这里我们维护两个变量,一个是P (Previous),表示手头已有的,还没有被编码的字符串,一个是C (current),表示当前新读进来的字符。下图展示了其编码过程。
在这里插入图片描述

  1. 初始状态,字典里只有所有的默认项,例如0->a,1->b,2->c。此时P和C都是空的。
  2. 读入新的字符C,与P合并形成字符串P+C。
  3. 在字典里查找P+C,如果:
    P+C在字典里,P=P+C。
    P+C不在字典里,将P的记号输出;在字典中为P+C建立一个记号映射;更新P=C。
  4. 返回步骤2重复,直至读完原字符串中所有字符。

3.2 举例

假设现在有数据字典如下:

字符编码
a0
b1
c2

现在将ababcababac(11个字符)进行编码。具体步骤如下:

步骤PCP+CP+C in Dictactionoutput
1-aaYES更新P=a-
2ababNO添加数据字典3->ab,更新P=b0
3babaNO添加数据字典4->ba,更新P=a1
4ababYES更新P=ab-
5abcabcNO添加数据字典5->abc,更新P=c3
6cacaNO添加数据字典6->ca,更新P=a2
7ababYES更新P=ab-
8abaabaNO添加数据字典7->aba,更新P=a3
9ababYES更新P=ab-
10abaabaYES更新P=aba-
11abacabacNO添加数据字典8->abac,更新P=c7
12c----2

所以,最终编码为0132372(7个字符)。

4. LZW解码算法

4.1 解码概述

解码的过程比编码复杂,其核心思想在于解码需要还原出编码时的用的字典。因此要理解解码的原理,必须分析它是如何对应编码的过程的。下面首先给出算法:

解码器的输入是压缩后的数据,即记号流 (Symbol Stream)。类似于编码,我们仍然维护两个变量pW (previous word) 和cW (current word),实际上就是记号 (Symbol),或者说子串。pW表示之前刚刚解码的记号;cW表示当前新读进来的记号。
在这里插入图片描述

  1. 初始状态,字典里只有所有的默认项,例如0->a,1->b,2->c。此时pW和cW都是空的。

  2. 读入第一个的符号cW,解码输出。注意第一个cW肯定是能直接解码的,而且一定是单个字符。

  3. 赋值P=C,pW=cW。

  4. 读入下一个符号cW。

  5. 在字典里查找cW,如果:

    1. cW在字典里:
      (1) 解码cW的值为C,即输出C。
      (2) 令P+C的第一个字符添加到记号映射。
      (3) 更新P = C。

    2. cW不在字典里:
      (1) C=P的第一个字符
      (2) 在字典中为P+C添加新的记号映射。

      (3) 输出P+C。

      (4) 更新P = P + C

  6. 返回步骤3重复,直至读完所有记号。

4.2 举例

我们那上面编码0132372的结果进行解码。初始化字典任然为如下表:

字符编码
a0
b1
c2

首先读取第一个记号cW=0,解码为a,输出,赋值pW=cW=0。然后开始循环,依此读取后面的记号:

StepPWCWCW In DictPCactionoutput
1-0YES-aPW=CW=0;P=C=a;a
201YESabPW=CW=1;P=C=b;3->ab(P+C第一个字符)b
313YESbabPW=CW=3;P=C=b;4->ba(P+C第一个字符)ab
432YESabcPW=CW=2;P=C=c;5->abc(P+C第一个字符)c
523YEScabPW=CW=3;P=C=ab;6->ca(P+C第一个字符)ab
637NOab-C=a(P第一个字母);输出aba(P+C);7->aba(P+C);PW=CW=7;P=abaaba
772YESabacPW=CW=2;P=C=c;3->abac(P+C第一个字符)c

因此,最终解码的字符为:ababcababac

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

surpassLiang

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值