- 深入理解序列化与反序列化
- 潘洪安
- 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=1739204853-8HjucAYuAhYt8DNZrgB5KJwmuyxfzYCh-0-97c63979a548a33cab59aad5969431e2)
以整数-1为例,最高位移位操作如表1-13所示。
表1-13 整数-1的最高位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_25.jpg?sign=1739204853-rJnHoIy2GFyiBiX2y31eUVW9sNPTTSvU-0-7430b7801d57dd98073bed6ab90b8312)
从表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=1739204853-0BDtlO1MZh8SD0JD26h5hJwbZ6HgrWqa-0-0dfd224caaa0b954b373271fc95f1bb6)
以整数-1为例,ZigZag编码的过程如表1-15所示。
表1-15 整数-1的ZigZag编码过程
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_27.jpg?sign=1739204853-qlHBkwl9ThBxW7RarkOWd4wpfdLYEmDL-0-298dc1fba58ab6de8f08dae86f91ba0c)
0的ZigZag编码和补码一致,读者可自行验证。
至此,正整数、0、负整数都可以用ZigZag编码来表示了。
1.5.2 ZigZag编码算法实现
ZigZag编码算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_28.jpg?sign=1739204853-t7uDChdaKLpzq8pLfO4fELenZXHGbuDh-0-adf1a1893c964f2a0c92dd8abe9e0bc1)
上述代码看起来并不太好理解,下面通过步骤分解的方式来分析代码要表达的意思,如表1-16所示。
表1-16 ZigZag编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_29.jpg?sign=1739204853-2I8AgH7zqmcZrY1uGj31pMxx9FauWlBn-0-5352cd9bea5dbbae58f5e54a67541f42)
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=1739204853-zPAEagseOZ0yCv8XTBx2YDtvbXw4OVXP-0-b3f379ce8299c1b6de13923e05bdca89)
以整数-1为例,ZigZag反编码如表1-18所示。
表1-18 整数-1的ZigZag反编码
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_31.jpg?sign=1739204853-qBz7IwAW1Zy82vzG9gpkcxh9j2weH8xL-0-8bd00cd8ea20ff2865d7c3ff58908a27)
2)将整数补码最低位移到最高位。以整数1为例,最低位移位操作如表1-19所示。
表1-19 整数1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_32.jpg?sign=1739204853-qst5NSW86CJ2ueS0LX6hGnJoAmhvnOgR-0-87068ee106712bb3789bbf4ea335c6ad)
以整数-1为例,最低位移位操作如表1-20所示。
表1-20 整数-1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_33.jpg?sign=1739204853-9akQ83DAr0xEaUE3J3BPlmVRDzNryX8G-0-f29944f0e52293de6d2709dfeb5a9100)
1.5.4 ZigZag反编码算法实现
ZigZag编码还原为整数的算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_34.jpg?sign=1739204853-C7TSXjI9hTgoknr5AOkSRDrYJ3xb3Cfr-0-4df4e98d7d83a94e4caf05d093d474a5)
上述代码看起来也不太好理解,依然通过步骤分解的方式来分析代码要表达的意思,如表1-21所示。
表1-21 ZigZag反编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_35.jpg?sign=1739204853-3PKa4cZ8kUqUvtXRjqivZpIBZJ9p4IbE-0-a6111db2ac518fe71d8cc19b998323ca)
1.5.5 总结
ZigZag编码使用的前提是:在大多数情况下使用的数字都是小整数,比如用户年龄、班级、年级、购物数量等。当数字比较大的时候,需要5字节来表示整数。ZigZag编码机制被用于Thrift、Protocol Buffer、Avro等序列化方案中。