- 深入理解序列化与反序列化
- 潘洪安
- 930字
- 2020-11-21 19:41:45
1.5 ZigZag编码
1.5.1 ZigZag编码流程
ZigZag将有符号整数统一映射为无符号整数,再通过Varint编码规则达到数据压缩的效果。ZigZag的编码流程如下:
1)将整数补码最高位移到最低位。正整数的最高位为0,数值越小,高位的0越多。负整数的最高位为1,绝对值越小,高位的1越多。以整数1为例,最高位移位操作如表1-12所示。
表1-12 整数1的最高位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_24.jpg?sign=1739168090-0hfNHnyumQz92TwJHtpa7JqyO8OXIdRO-0-cce81306d04fe0f63ff44c54f62ef21c)
以整数-1为例,最高位移位操作如表1-13所示。
表1-13 整数-1的最高位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_25.jpg?sign=1739168090-n8apVhn8sQNeZaP2aJwfzCFpbpBQ7XRF-0-ea4af4b10d665b75be53cbb77549181b)
从表1-13可以看出,负数绝对值越小,包含的前导1越多,用Varint编码压缩效果越差。
2)如果是负数,除符号位外,其他位取反;正数无须操作。这一步获得的便是ZigZag编码。以整数1为例,ZigZag编码的过程如表1-14所示。
表1-14 整数1的ZigZag编码过程
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_26.jpg?sign=1739168090-Pjpn8aJQFoHPA7XeZW8eS1srVNAO2slI-0-08de2b959fd25a63889f05c3fb8cdc24)
以整数-1为例,ZigZag编码的过程如表1-15所示。
表1-15 整数-1的ZigZag编码过程
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_27.jpg?sign=1739168090-rkDM9HQ4Vs7W2eSWaBhyrudGCIpCol12-0-468db21fb86ea84489c0ec85ccd61c33)
0的ZigZag编码和补码一致,读者可自行验证。
至此,正整数、0、负整数都可以用ZigZag编码来表示了。
1.5.2 ZigZag编码算法实现
ZigZag编码算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_28.jpg?sign=1739168090-dSQKnOayAXJT53MzX8eGhN2umBCS6CGa-0-2710718862929f1d5a6f865ab0d2c13d)
上述代码看起来并不太好理解,下面通过步骤分解的方式来分析代码要表达的意思,如表1-16所示。
表1-16 ZigZag编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_29.jpg?sign=1739168090-mOqpXh6KpeTzpDHCmOfuJyMQNgK7tGfC-0-d9222f6e0050289b9f71ae4d2dcebb1e)
1)n << 1:表示将整个值左移1位,正数、0、负数的最后1位就变成了0。
2)n >> 31:符号位放到最后1位。如果是非负数,则为全0;如果是负数,就是全1。
3)异或操作后可以看到,数据位全部反转了,而符号位保持不变,且移动到了最后1位。
1.5.3 ZigZag反编码流程
ZigZag反编码流程如下:
1)如果最低位是1,表示是负数,除符号位外,其他位取反;如果最低位是0,表示是正数,无须操作。以整数1为例,ZigZag反编码如表1-17所示。
表1-17 整数1的ZigZag反编码
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_30.jpg?sign=1739168090-uCZ2p39wDiMg7fIMLvq11pRXsMz09jsV-0-f97f0a43c93dabbd939f53fee7386171)
以整数-1为例,ZigZag反编码如表1-18所示。
表1-18 整数-1的ZigZag反编码
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_31.jpg?sign=1739168090-9T9BJ4QZCDwuMqIDVCfDf2Dl5pbDI8IH-0-4095214d7310fd8523c984037b4592bc)
2)将整数补码最低位移到最高位。以整数1为例,最低位移位操作如表1-19所示。
表1-19 整数1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_32.jpg?sign=1739168090-aZKJxYCPhdDC2aS30XjZFzl2TxD3Qb79-0-6fdfd955d778d006db1e6c1114031f71)
以整数-1为例,最低位移位操作如表1-20所示。
表1-20 整数-1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_33.jpg?sign=1739168090-1dzRUQnTOwyl4qSBb4IMVBLBfYnN5v4K-0-fc3687abc26e0e01974a3ee24a980fee)
1.5.4 ZigZag反编码算法实现
ZigZag编码还原为整数的算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_34.jpg?sign=1739168090-EuIa0m0vzXsSizlGynSGwvm2u13FkS8N-0-dfc05bc7189b25ee034d612a878d2b1a)
上述代码看起来也不太好理解,依然通过步骤分解的方式来分析代码要表达的意思,如表1-21所示。
表1-21 ZigZag反编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_35.jpg?sign=1739168090-h1xl5ZOp03GZvc0C51crWqItlzw2DAg2-0-6d163063e009d7bee6ec261fc489ebbf)
1.5.5 总结
ZigZag编码使用的前提是:在大多数情况下使用的数字都是小整数,比如用户年龄、班级、年级、购物数量等。当数字比较大的时候,需要5字节来表示整数。ZigZag编码机制被用于Thrift、Protocol Buffer、Avro等序列化方案中。