golang常用库之- encoding/binary包 | 字节转换成整形整形转换成字节、“大字端” 和 “小字端”

背景使用场景

Go编程三十六个套路: int与[]byte互转用于数据传输
参考URL: https://blog.frognew.com/2016/03/goguideencodingbinary.html

在用Go进行数据传输场景下,例如文件传输文件存储时,需要将Go数据例如int转换为[]byte。 得到的[]byte可以一步网络传输写入文件中。这个场景需要借助go标准库中encoding/binary包来实现

变长值是使用一到多个字节编码整数方法,绝对值较小的数字占用较少的字节数。这个ProtocolBuffer编码文档中有详细说明

encoding/binary相对效率更注重简单。 如果需要高效的序列化,特别是数据结构复杂的,可以选择高级解决方案例如encoding/gob包(Go语言自带数据编码解码工具包),或者采用ProtocolBuffer(跨语言)。

使用encoding/binary可以实现序列化和反序列化功能

使用encoding/binary包的优缺点

如果是go语言之间的序列化和反序列化推荐使用encoding/gob包,跨语言的序列化和反序列化可以使用protobuf,使用protobuf的化我们就不用考虑socket接受大小端的事情,protobuf数据里面就做了这种事情。

go语言socket通信大小转换问题

一般来说网络传输的字节序,可能是大端序或者小端序,取决于软件开始时通讯双方的协议规定。TCP/IP协议RFC1700规定使用“大端”字节序为网络字节序,开发时候需要遵守这一规则默认golang是使用大端序。详情golang中包encoding/binary已提供了大、小端序的使用。

内存中这些字节是按照从大到小的地址空间存储还是从小到大发送接收双方事先约定好,否则就会不同的顺寻着对接数据解析顺序不同出错发送端和解析端必须一致!

encoding/binary包

官方https://pkg.go.dev/encoding/binary

golangbinary包简单实现数字number)到字节序(byte sequences)的转换,以及64位整型varint)的编码与解码

Go语言自定义二进制文件的读写操作

Go语言自定义二进制文件的读写操作
参考URL: http://c.biancheng.net/view/4570.html

Go语言的 encoding/binary 包中的 binary.Write() 函数使得以二进制格式写数据非常简单,函数原型如下

func Write(w io.Writer, order ByteOrder, data interface{}) error

Write 函数可以将参数 data 的 binary 编码格式入参数 w 中,参数 data 必须是定长值、定长值的切片、定长值的指针参数 order 指定写入数据的字节序,写入结构体时,名字中有_字段会置为 0

package main

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

type Website struct {
    Url int32
}

func main() {
    file, err := os.Create("output.bin")
    for i := 1; i <= 10; i++ {
        info := Website{
            int32(i),
        }
        if err != nil {
            fmt.Println("文件创建失败 ", err.Error())
            return
        }
        defer file.Close()

        var bin_buf bytes.Buffer
        binary.Write(&amp;bin_buf, binary.LittleEndian, info)
        b := bin_buf.Bytes()
        _, err = file.Write(b)

        if err != nil {
            fmt.Println("编码失败", err.Error())
            return
        }
    }
    fmt.Println("编码成功")
}

运行上面的程序会在当前目录生成 output.bin 文件,文件内容如下:

0100 0000 0200 0000 0300 0000 0400 0000
0500 0000 0600 0000 0700 0000 0800 0000
0900 0000 0a00 0000 

字节转换成整形整形转换成字节

使用 Go 语言中 binary 这个标准包,该包实现了数字与字节之间的转化。

下来我们数字 0x22334455 转化为大字端字节存储

buffer := new(bytes.Buffer) 
binary.Write(buffer, binary.BigEndian, int32(0x22334455)) 

binary.BigEndian 常量表示大字端。binary.LittleEndian 常量表示小字端。

代码demo

package main

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

//字节转换整形
func BytesToInt(b []byte) int {
	bytesBuffer := bytes.NewBuffer(b)

	var x int32
	binary.Read(bytesBuffer, binary.BigEndian, &amp;x)  //转换有两种不同方式,也就是大端和小端。大端就是内存中低地址对应整数的高位。

	return int(x)
}

//整形转换成字节
func IntToBytes(n int) []byte {
	x := int32(n)

	bytesBuffer := bytes.NewBuffer([]byte{})
	binary.Write(bytesBuffer, binary.BigEndian, x)
	return bytesBuffer.Bytes()
}

func main() {

	fmt.Println(IntToBytes(2))
	fmt.Println(len(IntToBytes(2)))

}

小字端” 和 “大字端”

计算机字节序和网络字节序

字节序 就是多字节数据类型 (int, float 等)在内存中的存储顺序。可分为大端序,低地址端存放高位字节;小端序与之相反,低地址端存放低位字节。

  • 小端字节序(Little Endian): 低位字节排放在内存的低地址端,高位字节排放在内存的高地址端
  • 大端字节序(Big Endian): 高位字节排放在内存的低地址端,低位字节排放在内存的高地址端

在这里插入图片描述
使用小端序时不移动字节就能改变 number 占内存的大小而不需内存地址起始位。比如我想把四字节的 int32 类型整型转变为八字节的 int64 整型,只需在小端序末端加零即可

44 33 22 11
44 33 22 11 00 00 00 00
上述扩展缩小整型变量操作编译器层面非常有用,但在网络协议层非也。

例如,对于一个加法器,选择的是小字端。为什么
因为,加法是从低位到高位开始加,一旦有进位,就直接送到下一位,设计就很简单。

在进行数据传输需要考虑一下字节序,因为不同处理器架构体系会使用不同的存储字节序,而对于TCP/IP网络传输的字节序则固定采用的是大端字节序。

对于网络传输,使用的就是大字端。为什么

因为,早年设备缓存很小,先接收高字节能快速判断报文信息:包长度(需要准备多大缓存)、地址范围(IP地址是从前到后匹配的)。

性能不是很好的设备上,高字节在先确实是会更快一些。

TCP/IP协议RFC1700规定使用“大端”字节序为网络字节序,开发时候需要遵守这一规则~

网络协议操作二进制数字时约定使用大端序,大端序是网络字节传输采用的方式

原文地址:https://blog.csdn.net/inthat/article/details/129141743

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任

如若转载,请注明出处:http://www.7code.cn/show_34982.html

如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱suwngjj01@126.com进行投诉反馈,一经查实,立即删除

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注