首页 > 后端开发 > Golang > 正文

Go语言中int16到字节数组的转换:encoding/binary包详解

DDD
发布: 2025-10-17 12:16:01
原创
522人浏览过

Go语言中int16到字节数组的转换:encoding/binary包详解

本文详细阐述go语言中将`int16`类型转换为2字节数组的标准方法,主要通过`encoding/binary`包实现。文章将介绍`putuint16`和`binary.write`两种函数的使用,并强调大小端序的重要性,确保数据转换的准确性和可靠性。

在Go语言中,将一个int16类型的整数转换为一个长度为2的字节数组([]byte)是常见的操作,尤其是在网络通信或文件存储等场景中。直接通过类型转换如[]byte(string(i))是错误的做法,因为这将把整数解释为Unicode码点并尝试转换为UTF-8字符串,这与我们期望的二进制表示完全不符。Go标准库提供了encoding/binary包,专门用于处理这种固定大小整数与字节序列之间的转换,它提供了高效且可靠的解决方案。

使用binary.PutUint16进行字节切片转换

当需要将int16值写入到一个预先分配好的字节切片时,binary.PutUint16函数是首选。此函数需要指定字节序(Endianness),这决定了多字节数据在内存或传输中的存储顺序。

基本用法:

package main

import (
    "encoding/binary"
    "fmt"
)

func main() {
    var i int16 = 41 // 待转换的int16值
    b := make([]byte, 2) // 创建一个长度为2的字节切片

    // 使用LittleEndian(小端序)将uint16写入字节切片
    // 注意:PutUint16接受uint16类型,因此需要将int16强制转换为uint16
    binary.LittleEndian.PutUint16(b, uint16(i))
    fmt.Printf("int16值 %d 转换为小端序字节数组: %v\n", i, b) // 输出: int16值 41 转换为小端序字节数组: [41 0]

    // 转换为大端序示例
    bBigEndian := make([]byte, 2)
    binary.BigEndian.PutUint16(bBigEndian, uint16(i))
    fmt.Printf("int16值 %d 转换为大端序字节数组: %v\n", i, bBigEndian) // 输出: int16值 41 转换为大端序字节数组: [0 41]

    // 验证转换回来的值
    retrievedVal := int16(binary.LittleEndian.Uint16(b))
    fmt.Printf("从字节数组 %v 恢复的int16值: %d\n", b, retrievedVal) // 输出: 从字节数组 [41 0] 恢复的int16值: 41
}
登录后复制

代码解析:

立即学习go语言免费学习笔记(深入)”;

  1. var i int16 = 41: 定义一个int16变量。
  2. b := make([]byte, 2): 创建一个长度为2的字节切片,用于存放转换结果。
  3. binary.LittleEndian.PutUint16(b, uint16(i)): 这是核心转换操作。
    • binary.LittleEndian:指定使用小端序。在小端序中,低位字节存储在内存的低地址,高位字节存储在高地址。对于41 (0x0029),小端序表示为 [0x29, 0x00]。
    • binary.BigEndian:指定使用大端序。在大端序中,高位字节存储在内存的低地址,低位字节存储在高地址。对于41 (0x0029),大端序表示为 [0x00, 0x29]。
    • uint16(i):PutUint16函数要求输入uint16类型。由于int16和uint16都是16位整数,它们在二进制表示上是兼容的,因此可以直接进行类型转换。如果int16为负数,转换成uint16会保留其二进制补码表示。

使用binary.Write进行流式写入

如果你的目标不是一个简单的字节切片,而是一个io.Writer接口(例如,网络连接、文件句柄等),那么binary.Write函数会更加方便。它能直接将固定大小的整数写入到流中,并自动处理字节序。

微软文字转语音
微软文字转语音

微软文本转语音,支持选择多种语音风格,可调节语速。

微软文字转语音 0
查看详情 微软文字转语音

基本用法:

package main

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

func main() {
    var i int16 = 41 // 待转换的int16值
    buf := new(bytes.Buffer) // 使用bytes.Buffer作为io.Writer的实现

    // 将int16值以小端序写入到buf中
    err := binary.Write(buf, binary.LittleEndian, i)
    if err != nil {
        fmt.Println("写入失败:", err)
        return
    }
    fmt.Printf("int16值 %d 以小端序写入流后字节数组: %v\n", i, buf.Bytes()) // 输出: int16值 41 以小端序写入流后字节数组: [41 0]

    // 转换为大端序示例
    bufBigEndian := new(bytes.Buffer)
    err = binary.Write(bufBigEndian, binary.BigEndian, i)
    if err != nil {
        fmt.Println("写入失败:", err)
        return
    }
    fmt.Printf("int16值 %d 以大端序写入流后字节数组: %v\n", i, bufBigEndian.Bytes()) // 输出: int16值 41 以大端序写入流后字节数组: [0 41]
}
登录后复制

代码解析:

立即学习go语言免费学习笔记(深入)”;

  1. buf := new(bytes.Buffer): 创建一个bytes.Buffer实例,它实现了io.Writer接口,可以作为流的替代品进行演示。
  2. err := binary.Write(buf, binary.LittleEndian, i):
    • buf: 目标io.Writer。
    • binary.LittleEndian:指定字节序。
    • i: 要写入的int16变量。binary.Write能够直接处理int16、uint16、int32等固定大小的整数类型,无需手动类型转换。
  3. if err != nil: binary.Write可能会返回错误,因此需要进行错误处理。

注意事项

  • 字节序(Endianness):这是最关键的因素。发送方和接收方必须使用相同的字节序来解释数据,否则会导致数据解析错误。常见的有小端序(Little Endian)和大端序(Big Endian)。网络协议通常使用大端序(网络字节序)。
  • 类型匹配:encoding/binary包提供了针对所有固定大小整数类型(如int8, uint8, int16, uint16, int32, uint32, int64, uint64)的PutXxx和Xxx(用于读取)函数,以及通用的Write和Read函数。确保选择与你的数据类型匹配的函数。
  • 错误处理:在使用binary.Write或binary.Read时,务必检查返回的错误,以确保数据操作的成功。
  • 负数处理:int16的负数会以其二进制补码形式进行转换。例如,int16(-1)在二进制补码中是0xFFFF。使用binary.LittleEndian.PutUint16转换为字节数组时,它会按0xFFFF的字节表示进行存储。在恢复时,如果使用binary.LittleEndian.Uint16得到uint16值,再将其强制转换为int16,会正确地恢复为-1。

总结

encoding/binary包是Go语言中处理整数与字节数组之间转换的标准且推荐的方式。它提供了灵活的函数来适应不同的场景,无论是直接操作字节切片还是通过io.Writer进行流式写入。理解并正确应用字节序是确保数据正确解析的关键。通过本文的示例和说明,开发者可以自信地在Go项目中实现int16到字节数组的准确转换。

以上就是Go语言中int16到字节数组的转换:encoding/binary包详解的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号