Mike Fernandez
Mike Fernandez

Reputation: 471

golang: convert uint32 (or any built-in type) to []byte (to be written in a file)

I'm trying to convert an uint32 to a byte array (4 bytes) in Go using the unsafe library:

h := (uint32)(((fh.year*100+fh.month)*100+fh.day)*100 + fh.h)
a := make([]byte, unsafe.Sizeof(h))
copy(a, *(*[]byte)(unsafe.Pointer(&h)))

The first two lines are correct, but then I get a runtime error ( unexpected fault address ) at the copy call.

The next step would be to call Write

_, err = fi.Write(a)

to write the 4 bytes into a file.

I've found other questions with a similar topic, but none with a working code. I'm also aware that unsafe is unsafe.

Any help would be greatly appreciated.

Upvotes: 20

Views: 29444

Answers (1)

Thundercat
Thundercat

Reputation: 120941

Avoid the unsafe package.

Use the encoding/binary package to convert a uint32 to a slice of bytes:

h := (uint32)(((fh.year*100+fh.month)*100+fh.day)*100 + fh.h)
a := make([]byte, 4)
binary.LittleEndian.PutUint32(a, h)
_, err = fi.Write(a)

This one-liner does the same thing, but has an additional runtime cost:

err := binary.Write(fi, binary.LittleEndian, (uint32)(((fh.year*100+fh.month)*100+fh.day)*100 + fh.h))

Here's how to do the conversion with the unsafe package:

h := (uint32)(((fh.year*100+fh.month)*100+fh.day)*100 + fh.h)
a := (*[4]byte)(unsafe.Pointer(&h))[:]
_, err = fi.Write(a)

The expression (*[4]byte)(unsafe.Pointer(&h)) converts a uint32 pointer to a [4]byte pointer. The [:] at the end creates a slice on the [4]byte.

The code in the question interprets the uint32 as a slice header. The resulting slice is not valid and copy faults.

Upvotes: 22

Related Questions