golang 结构体 字节对齐是怎么样的

Python027

golang 结构体 字节对齐是怎么样的,第1张

用golang解析二进制协议时,其实没必要管结构体的字段的对齐规则,何况语言规范也没有规定如何对齐,也就是没有规则。用encoding/binary.Read函数直接读入struct里就行,struct就像c那样写

type Data struct {

Size, MsgType uint16

Sequence uint32

// ...

}

golang编译器加不加padding,Read都能正常工作,runtime知道Data的布局的,不像C直接做cast所以要知道怎样对齐。

用unsafe.Alignof可以知道每个field的对齐长度,但没必要用到。

package main

/*

#include <stdint.h>

#pragma pack(push, 1)

typedef struct {

uint16_t size

uint16_t msgtype

uint32_t sequnce

uint8_t data1

uint32_t data2

uint16_t data3

} mydata

#pragma pack(pop)

mydata foo = {

1, 2, 3, 4, 5, 6,

}

int size() {

return sizeof(mydata)

}

*/

import "C"

import (

"bytes"

"encoding/binary"

"fmt"

"log"

"unsafe"

)

func main() {

bs := C.GoBytes(unsafe.Pointer(&C.foo), C.size())

fmt.Printf("len %d data %v\n", len(bs), bs)

var data struct {

Size, Msytype uint16

Sequence uint32

Data1 uint8

Data2 uint32

Data3 uint16

}

err := binary.Read(bytes.NewReader(bs), binary.LittleEndian, &data)

if err != nil {

log.Fatal(err)

}

fmt.Printf("%v\n", data) // {1 2 3 4 5 6}

buf := new(bytes.Buffer)

binary.Write(buf, binary.BigEndian, data)

fmt.Printf("%d %v\n", buf.Len(), buf.Bytes()) // 15 [0 1 0 2 0 0 0 3 4 0 0 0 5 0 6]

}

int is a signed integer type that is at least 32 bits in size. It is a distinct type, however, and not an alias for, say, int32.

int 是带符号整数类型,其大小至少为32位。 它是一种确切的类型,而不是 int32 的别名。

int 不是int32,那 int 在内存站多少字节呢?官方没有明确,让我们测试下。

GOARCH="amd64"

package main

import (

"fmt"

"unsafe"

)

func main() {

i := int(1)

fmt.Println(unsafe.Sizeof(i)) // 4

j := 1

fmt.Println(unsafe.Sizeof(j)) // 4

u := uint(1)

fmt.Println(unsafe.Sizeof(u)) // 4

}

可以认为 int 是4字节么?我不敢这样认为,GoLang支持多种平台架构。如果对size有明确要求,那就用 int32 之类的吧。

补充:随Go版本的变化,这个的确是在变化,所以到底占用多少字节,还是看具体版本吧