登录
首页 >  Golang >  Go问答

使用 Golang 对文件进行二进制编码/解码并计算不同的校验和

来源:stackoverflow

时间:2024-03-06 20:45:21 145浏览 收藏

积累知识,胜过积蓄金银!毕竟在Golang开发的过程中,会遇到各种各样的问题,往往都是一些细节知识点还没有掌握好而导致的,因此基础知识点的积累是很重要的。下面本文《使用 Golang 对文件进行二进制编码/解码并计算不同的校验和》,就带大家讲解一下知识点,若是你对本文感兴趣,或者是想搞懂其中某个知识点,就请你继续往下看吧~

问题内容

我正在用 golang 编码和解码文件。我特别需要我正在使用的二维数组,这只是测试代码来说明这一点。我不完全确定我做错了什么,我试图将文件转换为 uint32 数字列表,然后获取这些数字并将它们转换回文件。问题是,当我这样做时,文件看起来不错,但校验和不对齐。我怀疑我在转换为 uint32 时做错了什么。我必须执行 switch/case,因为我无法知道在给定文件末尾我将读取多少字节。

package main

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

const (
    headerSeq = 8
    body      = 24
)

type part struct {
    Seq  int
    Data uint32
}

func main() {
    f, err := os.Open("speech.pdf")
    if err != nil {
        panic(err)
    }
    defer f.Close()

    reader := bufio.NewReader(f)
    b := make([]byte, 4)
    o := make([][]byte, 0)
    var value uint32
    for {
        n, err := reader.Read(b)
        if err != nil {
            if err != io.EOF {
                panic(err)
            }
        }

        if n == 0 {
            break
        }
        fmt.Printf("len array %d\n", len(b))
        fmt.Printf("len n %d\n", n)
        switch n {
        case 1:
            value = uint32(b[0])
        case 2:
            value = uint32(uint32(b[1]) | uint32(b[0])<<8)
        case 3:
            value = uint32(uint32(b[2]) | uint32(b[1])<<8 | uint32(b[0])<<16)
        case 4:
            value = uint32(uint32(b[3]) | uint32(b[2])<<8 | uint32(b[1])<<16 | uint32(b[0])<<24)
        }
        fmt.Println(value)
        bs := make([]byte, 4)
        binary.BigEndian.PutUint32(bs, value)
        o = append(o, bs)
    }
    fo, err := os.OpenFile("test.pdf", os.O_APPEND|os.O_WRONLY|os.O_CREATE, 0600)
    if err != nil {
        panic(err)
    }
    defer fo.Close()

    for _, ba := range o {
        _, err := fo.Write(ba)
        if err != nil {
            panic(err)
        }
    }
}

正确答案


因此,您想在文件中写入和读取不同长度的数组。

import "encoding/binary"

// You need a consistent byte order for reading and writing multi-byte data types
const order = binary.LittleEndian

var dataToWrite = []byte{ ... ... ... }
var err error

// To write a recoverable array of varying length
var w io.Writer
// First, encode the length of data that will be written
err = binary.Write(w, order, int64(len(dataToWrite)))
// Check error
err = binary.Write(w, order, dataToWrite)
// Check error

// To read a variable length array
var r io.Reader
var dataLen int64
// First, we need to know the length of data to be read
err = binary.Read(r, order, &dataLen)
// Check error
// Allocate a slice to hold the expected amount of data
dataReadIn := make([]byte, dataLen)
err = binary.Read(r, order, dataReadIn)
// Check error

此模式不仅适用于 byte,还适用于任何其他固定大小的数据类型。有关编码的详细信息,请参阅 binary.Write

如果编码数据的大小是一个大问题,您可以通过将数组长度存储为 varint binary.PutVarintbinary.ReadVarint 来节省一些字节

以上就是《使用 Golang 对文件进行二进制编码/解码并计算不同的校验和》的详细内容,更多关于的资料请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>