如何在 Go 中将整数写入字节切片的指定位置

本文讲解如何将一个大于 255 的整数(如 300)以多字节形式(如 uint16)写入 go 字节切片的指定索引区间,重点解决 `[]byte` 不支持直接赋值整数、需手动编码字节序等核心问题。

在 Go 中,[]byte 是字节切片,每个元素是 uint8 类型(取值范围 0–255),因此无法直接存储 300 这样的整数。若需将 300 写入连续两个字节(即 uint16),必须先将其按指定字节序(大端或小端)编码为两个 byte,再通过切片操作安全替换原数据。

最可靠的方式是使用 encoding/binary 包进行确定性序列化。以下是一个完整、可运行的示例:

package main

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

func intToBytesLE(i uint16) []byte {
    buf := new(bytes.Buffer)
    binary.Write(buf, binary.LittleEndian, i) // 小端:300 → [0x2c, 0x01]
    return buf.Bytes()
}

func intToBytesBE(i uint16) []byte {
    buf := new(bytes.Buffer)
    binary.Write(buf, binary.BigEndian, i) // 大端:300 → [0x01, 0x2c]
    return buf.Bytes()
}

func main() {
    app0 := []byte("\xff\xe0\x00\x10\x4a\x46\x49\x46\x00\x01\x01\x01\x00\x00\x00\x00\x00\x00")

    // 替换 app0[13:15](共 2 字节)为 uint16(300) 的小端表示
    replacement := intToBytesLE(300)
    app0 = append(app0[:13], append(replacement, app0[15:]...)...)

    fmt.Printf("Hex: %x\n", app0) // 输出中可见位置 13–14 变为 2c 01
}
✅ 关键说明: app0[13:15] 是长度为 2 的切片,恰好容纳 uint16 的两个字节; append(app0[:13], append(replacement, app0[15:]...)...) 是标准“切片拼接”模式:先截取前段 [0:13),再拼接新字节,最后追加后段 [15:]; 字节序必须明确选择:JPEG/XMP 等二进制格式通常采用 BigEndian(网络字节序),而 x86 系统本地内存多用 LittleEndian——请根据协议规范确认; 若需原地修改(避免内存重分配),且切片容量充足,也可直接用 binary.Write 写入底层数组: binary.LittleEndian.PutUint16(app0[13:], 300) // 更高效,无需新建切片

总结:Go 中整数到字节的转换不是隐式行为,必须显式编码。优先使用 binary.{Big,Little}Endian.PutUintXX() 进行原地写入(推荐),或配合 bytes.Buffer + append 实现灵活拼接。务必校验目标区间长度与整数类型字节数匹配(如 uint16 → 2 字节),并严格遵循协议约定的字节序。