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

Mike Fernandez picture Mike Fernandez · Mar 15, 2015 · Viewed 14.8k times · Source

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.

Answer

Cerise Limón picture Cerise Limón · Mar 15, 2015

Here's one way to do it:

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

Here's a breakdown of what's going on. The code

(*[4]byte)(unsafe.Pointer(&h))

converts the 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.

An alternative approach that does not use unsafe is to use the encoding/binary package:

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