![深入理解序列化与反序列化](https://wfqqreader-1252317822.image.myqcloud.com/cover/521/34667521/b_34667521.jpg)
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=1739933529-tMdiwMMU7WuBdqKjAT9XdIMBZn7qGTJP-0-cbdb27ca4b6774ca0fa6a6357cc16216)
以整数-1为例,最高位移位操作如表1-13所示。
表1-13 整数-1的最高位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_25.jpg?sign=1739933529-VhVfs0YIdNpWurAslmFCUnKo8QU1LWfW-0-15fddd94db90dea8d148a37ddb541ccd)
从表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=1739933529-2ZURzDCtKmScc27GoimFkLC5sjErfCS3-0-1fff9e3b055767be185665207d04850f)
以整数-1为例,ZigZag编码的过程如表1-15所示。
表1-15 整数-1的ZigZag编码过程
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_27.jpg?sign=1739933529-B7jmhusT2rfzE8gXNcaflhJphP7MVucL-0-f11c4176cb4c3bf5482de31181228b7c)
0的ZigZag编码和补码一致,读者可自行验证。
至此,正整数、0、负整数都可以用ZigZag编码来表示了。
1.5.2 ZigZag编码算法实现
ZigZag编码算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_28.jpg?sign=1739933529-CtCGk7hxMOh01aMBHMM9G6bn8RGfWxVU-0-b19917e414956b23765f08b6a75241ef)
上述代码看起来并不太好理解,下面通过步骤分解的方式来分析代码要表达的意思,如表1-16所示。
表1-16 ZigZag编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_29.jpg?sign=1739933529-PT3uN7mC5lY78TvL3k1txGMTZwmgI4wW-0-45a38c55d4a69a4ed804d72d465da421)
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=1739933529-T29SqQ0q11h3XHDzaOEec68anvbaUtnt-0-3df9fb2618a4f22e35726316877ac47e)
以整数-1为例,ZigZag反编码如表1-18所示。
表1-18 整数-1的ZigZag反编码
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_31.jpg?sign=1739933529-7kiNWUBW5faKx0PgAq4YOLjQNuNNdkro-0-91ad014c3bc6cae082cfab0a5648da0b)
2)将整数补码最低位移到最高位。以整数1为例,最低位移位操作如表1-19所示。
表1-19 整数1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_32.jpg?sign=1739933529-koDZYeIJGVNWrB1sqjPwdhRWoKGzfhkB-0-9ed6aff5b2d220bb5938341d3fc7e531)
以整数-1为例,最低位移位操作如表1-20所示。
表1-20 整数-1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_33.jpg?sign=1739933529-HFBaYHdI3QO8Q4wR0ui1nvxcfKDyEl1u-0-588b4752998827447f166c961655af85)
1.5.4 ZigZag反编码算法实现
ZigZag编码还原为整数的算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_34.jpg?sign=1739933529-1qIuIR7H6PAeaI6v02Q84YjBZslaDCq2-0-3b3cf5cee09f072552247566b4931e54)
上述代码看起来也不太好理解,依然通过步骤分解的方式来分析代码要表达的意思,如表1-21所示。
表1-21 ZigZag反编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_35.jpg?sign=1739933529-2urRTMRCmLdpcx4t5FRzmEWR0jYfm2Mw-0-b3573c179d5ca2899450a0f7c659e94c)
1.5.5 总结
ZigZag编码使用的前提是:在大多数情况下使用的数字都是小整数,比如用户年龄、班级、年级、购物数量等。当数字比较大的时候,需要5字节来表示整数。ZigZag编码机制被用于Thrift、Protocol Buffer、Avro等序列化方案中。