美文网首页
binary包快速入门

binary包快速入门

作者: 护念 | 来源:发表于2023-11-25 13:40 被阅读0次

binary

它的作用用一句话说就是:实现数据与二进制之间的双向转换

它有许多使用场景,比如:

  • 读取(解析)网络过来的二进制数据
  • 读取一个二进制文件
  • 将一个数据写成二进制

大小端字节序

开始之前先大概说下,大小端字节序,简单理解就是字节的编码顺序,一个从高位开始,一个从低位开始。

我们使用时候主要需要保证读和写的顺序保持一致就行,如果需要详细了解可以看这篇

1. 单一数据转换

好啦!我们开始来看最简单的,将一个简单数据,比如一个字符、一个数值转换成二进制

对于一些单一的字符读写可以使用,binary提供的binary.BigEndianbinary.LittleEndian完成。

package main

import (
    "encoding/binary"
    "fmt"
)

func main() {
    var a uint64 = 56

    // 用于存二进制数据
    bData := make([]byte, 8) // 64位 8个字节
  // a 写入到bData中
    binary.LittleEndian.PutUint64(bData, a)

    // 从bData读取 数据到b中
    b := binary.LittleEndian.Uint64(bData)

    fmt.Println(b)
}

// 56

2. 复杂(结构体)数据转换

需要注意的是,binary包处理的数据必须要有固定的长度,所以对结构体的字段类型有要求。

package main

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

// 需要注意的是在binary 出来的数据必须要有固定的长度
type Data struct {
    ID         uint16
    Temprature float32
    Records    [4]byte // 4个长度字节数组
}

func main() {
    // 准备好要转换的数据
    datas := []Data{
        {ID: 12, Temprature: 21.8, Records: [4]byte{'A', 'B', 'C', 'D'}},
        {ID: 2, Temprature: 28.8, Records: [4]byte{'B', 'B', 'C', 'A'}},
    }

    // 具有io.Writer 和 io.Reader的 缓冲对象用于存二进制数据
    var buf = new(bytes.Buffer)
    // 将数据写入buf中 记住这里最后一个参数必须是引用(比如指针,这里切片本身就是引用所以无所谓)
    binary.Write(buf, binary.LittleEndian, datas)

    parseData := make([]Data, 2) // 存解析出来的数据
    // 从buf中读取 解析成Data结构体切片
    binary.Read(buf, binary.LittleEndian, parseData)

    fmt.Printf("%#v\n", parseData[0])
    fmt.Printf("%#v\n", parseData[1])
    fmt.Printf("ID: %d , Temprature: %v, Record1: %c\n", parseData[0].ID, parseData[0].Temprature, parseData[0].Records[0])
}

// main.Data{ID:0xc, Temprature:21.8, Records:[4]uint8{0x41, 0x42, 0x43, 0x44}}
// main.Data{ID:0x2, Temprature:28.8, Records:[4]uint8{0x42, 0x42, 0x43, 0x41}}
// ID: 12 , Temprature: 21.8, Record1: A

3. 写二进制数据到文件

package main

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

func main() {
    file, err := os.OpenFile("data.bin", os.O_CREATE|os.O_RDWR, 0666)
    if err != nil {
        fmt.Println("创建文件错误")
        return
    }

    defer file.Close()

    // 封装一层缓冲区
    buf := bufio.NewWriter(file)
    // 写入缓冲中
    binary.Write(buf, binary.LittleEndian, []byte("hello world\n")) // 第一次写入 12个字节
    binary.Write(buf, binary.LittleEndian, []byte("beautiful!\n"))  // 第二次写入11字节

    // 从缓冲写入文件
    buf.Flush()

    // 从 文件中读取
    f, err := os.Open("data.bin")
    if err != nil {
        fmt.Println("打开文件失败:", err.Error())
        return
    }

    // 构造一个读bufio
    reader := bufio.NewReader(f)
    // 第一次读取 12个字节
    a := make([]byte, 12)
    binary.Read(reader, binary.LittleEndian, a)
    fmt.Printf("%s", a)
    // 第二次读取11个字节
    binary.Read(reader, binary.LittleEndian, a[0:11])
    fmt.Printf("%s", a[0:11])
}

// hello world
// beautiful!

4. 总结

本质上binary包只有两个操作: 一个写,一个读;分别对应两种转换:

  1. 写 - (数据->二进制)
  2. 读 - (二进制 -> 数据)

相关文章

网友评论

      本文标题:binary包快速入门

      本文链接:https://www.haomeiwen.com/subject/xldowdtx.html