![深入理解序列化与反序列化](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=1739297770-aXVLT806A6fAfeN417BtUHqtTaXHKFht-0-a52701ae6248bf8e4fbb7afdd29166cd)
以整数-1为例,最高位移位操作如表1-13所示。
表1-13 整数-1的最高位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_25.jpg?sign=1739297770-fdZnsGCcs3Sb9hDi47AYP6KDXXFR1JKV-0-307f4b009fff80e8f6b54d858149e254)
从表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=1739297770-d28SHRtee2FBVaBSL19Gq5k9ShhjcMGe-0-f946efc87839381f50b5c41739acc573)
以整数-1为例,ZigZag编码的过程如表1-15所示。
表1-15 整数-1的ZigZag编码过程
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_27.jpg?sign=1739297770-qQvEhDh3iuelbX1m0AWDQ21NB7b67hc7-0-3da43fa55f472feee9435a9947fcb4f8)
0的ZigZag编码和补码一致,读者可自行验证。
至此,正整数、0、负整数都可以用ZigZag编码来表示了。
1.5.2 ZigZag编码算法实现
ZigZag编码算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_28.jpg?sign=1739297770-lmAh0mKmuQD7DtuS8BnmAZrS4y1ielvb-0-9936946504d7454d8e0a31efce061539)
上述代码看起来并不太好理解,下面通过步骤分解的方式来分析代码要表达的意思,如表1-16所示。
表1-16 ZigZag编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_29.jpg?sign=1739297770-v6HT4UYD8xlrHA85sVIPTy4YRxIq5LpE-0-22d11a21bd2c8c6fada2c32a63262f68)
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=1739297770-DuHji4hiMVTdnl5fbHwI51yEqrwp2HuX-0-a60faf2704223d636fd668e4f07ee78d)
以整数-1为例,ZigZag反编码如表1-18所示。
表1-18 整数-1的ZigZag反编码
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_31.jpg?sign=1739297770-AZ7aDjXhzOpodriwEg0HgMsWTeavQqFn-0-b190912802c497807e5b161a3697968f)
2)将整数补码最低位移到最高位。以整数1为例,最低位移位操作如表1-19所示。
表1-19 整数1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_32.jpg?sign=1739297770-2rIuRv07e0Vmz2KZxjsjwSFnaU6ll262-0-531dd9c3de3bd709073151da81315404)
以整数-1为例,最低位移位操作如表1-20所示。
表1-20 整数-1的最低位移位操作
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_33.jpg?sign=1739297770-EFHaqyUxs0oXrQQIX1GEtT0pgqI1Hvaj-0-329f4142a712bb4efe2ef0b40208ac44)
1.5.4 ZigZag反编码算法实现
ZigZag编码还原为整数的算法实现如以下代码所示。
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_34.jpg?sign=1739297770-IFLlsFGaEBKIZm1bKno2Itp9JaIf3V69-0-df49fc0e09b10ab9aa33e36c8ca8bc4e)
上述代码看起来也不太好理解,依然通过步骤分解的方式来分析代码要表达的意思,如表1-21所示。
表1-21 ZigZag反编码算法步骤分解
![img](https://epubservercos.yuewen.com/171F30/18519308908425106/epubprivate/OEBPS/Images/txt002_35.jpg?sign=1739297770-kbc8OwQUKAsVk0cbJfBClj7MXHfKcCp0-0-c7db8c3d1ee1b340956c2b262116e0d1)
1.5.5 总结
ZigZag编码使用的前提是:在大多数情况下使用的数字都是小整数,比如用户年龄、班级、年级、购物数量等。当数字比较大的时候,需要5字节来表示整数。ZigZag编码机制被用于Thrift、Protocol Buffer、Avro等序列化方案中。