2022-06-13:golang中,[]byte和结构体如何相互转换?

2023-06-08 14:25:01 浏览数 (2)

2022-06-13:golang中,[]byte和结构体如何相互转换?

答案2022-06-13:

[]byte和结构体的转换的应用场景是数据解析。

代码里有两种方法,一种是内存不共用,另一种是内存共用。

[]byte转结构体严格将首地址需要是8的倍数。但代码里并没有遵守这个规则,测试后也没问题。

但有些场合需要严格遵守这个规则,否则会出现问题。

虽然这里没出现问题,但是结构体首地址最好是8的整数倍。不过很多CPU帮你解决了。MIPS会直接崩溃。

这里没出现问题,那是因为很多CPU帮你解决了。MIPS会直接崩溃。这个答案来自qq群友。我并没有去验证。

代码用golang编写。代码如下:

代码语言:javascript复制
package main

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

type T struct {
  A int64
  B float64
}

func main() {
  if true {
    fmt.Println("切片和结构体内存不共用")
    // Create a struct and write it.
    t := T{A: 1, B: 3.14}
    buf := &bytes.Buffer{}
    err := binary.Write(buf, binary.LittleEndian, t)
    if err != nil {
      panic(err)
    }
    fmt.Println(buf.Bytes())

    // Read into an empty struct.
    t = T{}
    err = binary.Read(buf, binary.LittleEndian, &t)
    if err != nil {
      panic(err)
    }
    fmt.Printf("%x %frn", t.A, t.B)
  }
  fmt.Println("-------------------------")
  if true {
    fmt.Println("切片和结构体内存共用")
    t1 := T{A: 1, B: 3.14}
    fmt.Println("原结构体:", t1)
    sm1 := SimulatedSlice{
      Addr: unsafe.Pointer(&t1),
      Len:  int(unsafe.Sizeof(t1)),
      Cap:  int(unsafe.Sizeof(t1)),
    }
    bytes1 := *(*[]byte)(unsafe.Pointer(&sm1))
    bytes1[0] = 2
    fmt.Println("修改切片,结构体变成:", t1)
    fmt.Println(bytes1)
    fmt.Println("")

    bytes2 := []byte{1, 3, 0, 0, 0, 0, 0, 0, 0, 31, 133, 235, 81, 184, 30, 9, 64}
    fmt.Println("原切片:", bytes2)
    t2 := (*T)(unsafe.Pointer(&bytes2[1]))
    t2.A = 4
    fmt.Println("以为会有内存对齐问题,实际上没问题。修改结构体,切片变成:", bytes2)
    fmt.Println(t2)
    fmt.Println("地址不是8的整数倍:", uintptr(unsafe.Pointer(&bytes2[1])))
    fmt.Println("地址不是8的整数倍:", uintptr(unsafe.Pointer(t2)))
    fmt.Println("严格将首地址需要是8的倍数。")
    fmt.Println("一般也没事。")
    fmt.Println("但有些场合需要严格遵守这个规则,否则会出现问题。")
    fmt.Println("虽然这里没出现问题,但是结构体首地址最好是8的整数倍。不过很多CPU帮你解决了。MIPS会直接崩溃。")
    fmt.Println("这里没出现问题,那是因为很多CPU帮你解决了。MIPS会直接崩溃。这个答案来自qq群友。我并没有去验证。")

  }
}

type SimulatedSlice struct {
  Addr unsafe.Pointer
  Len  int
  Cap  int
}

执行结果如下:

0 人点赞