登录
首页 >  Golang >  Go教程

Varint编码与二进制读取差异解析

时间:2026-01-25 23:54:56 253浏览 收藏

一分耕耘,一分收获!既然都打开这篇《Varint编码与二进制读取的区别解析》,就坚持看下去,学下去吧!本文主要会给大家讲到等等知识点,如果大家对本文有好的建议或者看到有不足之处,非常欢迎大家积极提出!在后续文章我会继续更新Golang相关的内容,希望对大家都有所帮助!

Go 中 Varint 编码与二进制字节读取的本质区别详解

`binary.Varint` 实现的是 Protocol Buffers 风格的变长整数编码(小端、7-bit 分块、MSB 标志位),而 `binary.Read` 是按指定字节序直接解析固定长度的原始二进制数据;二者语义完全不同,不可互换使用。

在 Go 标准库中,encoding/binary 包提供了两类截然不同的整数解析机制:定长二进制解析(如 binary.Read)和变长整数解码(binary.Varint)。它们面向完全不同的协议场景,混淆使用将导致严重逻辑错误。

? 本质差异解析

  • binary.Read(buf, order, &i)
    将缓冲区中连续的 8 字节(对 int64)按指定字节序(如 LittleEndian)直接解释为一个完整整数。它不关心内容含义,只做“裸字节到数值”的机械映射。
    在示例中:

    b := []byte{0x18, 0x2d, 0x44, 0x54, 0xfb, 0x21, 0x09, 0x40}
    // LittleEndian 解析:0x400921FB54442D18 → 十进制 4614256656552045848
  • binary.Varint(b)
    实现的是 Protocol Buffers 的 varint 编码规范

    • 每个字节仅用低 7 位存储数据,最高位(MSB)作为“是否继续”标志;
    • 数值按小端分组拼接(但不是字节序意义上的小端,而是 LSB 优先的 bit-level 编码);
    • 编码长度可变(1–10 字节),用于节省空间。
      示例中 b[0] = 0x18(二进制 00011000),MSB = 0 → 结束,有效 7 位为 0011000 = 12,因此返回 12。

✅ 正确使用示例

package main

import (
    "bytes"
    "encoding/binary"
    "fmt"
)

func main() {
    // 场景 1:读取固定长度的 int64(如文件头、网络包字段)
    raw := []byte{0x18, 0x2d, 0x44, 0x54, 0xfb, 0x21, 0x09, 0x40}
    var fixed int64
    binary.Read(bytes.NewReader(raw), binary.LittleEndian, &fixed)
    fmt.Printf("Fixed-size read: %d\n", fixed) // → 4614256656552045848

    // 场景 2:解码 protobuf-style varint(如 gRPC、.proto 序列化数据)
    varintBytes := []byte{0x18} // 0x18 = 0b00011000 → 7-bit payload 0b0011000 = 12
    v, n := binary.Varint(varintBytes)
    if n <= 0 {
        panic("invalid varint")
    }
    fmt.Printf("Varint decode: %d (consumed %d bytes)\n", v, n) // → 12
}

⚠️ 关键注意事项

  • ❌ binary.Varint 不接受 bytes.Buffer 或 io.Reader —— 它只接收 []byte 并从索引 0 开始解析,且仅解码第一个 varint,忽略后续字节;
  • ❌ 不要试图用 binary.Varint 解析 binary.Write 写出的数据(后者是定长原始字节);
  • ✅ 若需序列化 varint,应使用 binary.PutUvarint / binary.PutVarint,而非 binary.Write;
  • ? binary.Varint 返回的 n 表示实际消耗的字节数,务必校验 n > 0,避免静默失败。

理解并严格区分这两种机制,是正确处理二进制协议(如自定义通信格式、兼容 Protobuf 生态)的基础。选择依据始终是:你的数据源遵循哪种编码规范?

到这里,我们也就讲完了《Varint编码与二进制读取差异解析》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

前往漫画官网入口并下载 ➜
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>