README
¶
struc
Struc exists to pack and unpack C-style structures from bytes, which is useful for binary files and network protocols. It could be considered an alternative to encoding/binary
, which requires massive boilerplate for some similar operations.
Take a look at an example comparing struc
and encoding/binary
Struc considers usability first. That said, it does cache reflection data and aims to be competitive with encoding/binary
struct packing in every way, including performance.
Example struct
type Example struct {
Var int `struc:"int32,sizeof=Str"`
Str string
Weird []byte `struc:"[8]int64"`
Var []int `struc:"[]int32,little"`
}
Struct tag format
Var []int `struc:"[]int32,little,sizeof=StringField"`
will pack Var as a slice of little-endian int32, and link it as the size ofStringField
.sizeof=
: Indicates this field is a number used to track the length of a another field.sizeof
fields are automatically updated onPack()
based on the current length of the tracked field, and are used to size the target field duringUnpack()
.- Bare values will be parsed as type and endianness.
Endian formats
big
(default)little
Recognized types
pad
- this type ignores field contents and is backed by a[length]byte
containing nullsbool
byte
int8
,uint8
int16
,uint16
int32
,uint32
int64
,uint64
float32
float64
Types can be indicated as arrays/slices using []
syntax. Example: []int64
, [8]int32
.
Bare slice types (those with no [size]
) must have a linked Sizeof
field.
Private fields are ignored when packing and unpacking.
Example code
package main
import (
"bytes"
"github.com/lunixbochs/struc"
)
type Example struct {
A int `struc:"big"`
// B will be encoded/decoded as a 16-bit int (a "short")
// but is stored as a native int in the struct
B int `struc:"int16"`
// the sizeof key links a buffer's size to any int field
Size int `struc:"int8,little,sizeof=Str"`
Str string
// you can get freaky if you want
Str2 string `struc:"[5]int64"`
}
func main() {
var buf bytes.Buffer
t := &Example{1, 2, 0, "test", "test2"}
err := struc.Pack(&buf, t)
o := &Example{}
err = struc.Unpack(&buf, o)
}
Benchmark
BenchmarkEncode
uses struc. Stdlib
benchmarks use equivalent encoding/binary
code. Manual
encodes without any reflection, and should be considered an upper bound on performance (which generated code based on struc definitions should be able to achieve).
BenchmarkEncode 1000000 1265 ns/op
BenchmarkStdlibEncode 1000000 1855 ns/op
BenchmarkManualEncode 5000000 284 ns/op
BenchmarkDecode 1000000 1259 ns/op
BenchmarkStdlibDecode 1000000 1656 ns/op
BenchmarkManualDecode 20000000 89.0 ns/op
Documentation
¶
Index ¶
- func Pack(w io.Writer, data interface{}) error
- func PackWithOptions(w io.Writer, data interface{}, options *Options) error
- func PackWithOrder(w io.Writer, data interface{}, order binary.ByteOrder) error
- func Sizeof(data interface{}) (int, error)
- func SizeofWithOptions(data interface{}, options *Options) (int, error)
- func Unpack(r io.Reader, data interface{}) error
- func UnpackWithOptions(r io.Reader, data interface{}, options *Options) error
- func UnpackWithOrder(r io.Reader, data interface{}, order binary.ByteOrder) error
- type Custom
- type Field
- type Fields
- func (f Fields) Pack(buf []byte, val reflect.Value, options *Options) (int, error)
- func (f Fields) SetByteOrder(order binary.ByteOrder)
- func (f Fields) Sizeof(val reflect.Value, options *Options) int
- func (f Fields) String() string
- func (f Fields) Unpack(r io.Reader, val reflect.Value, options *Options) error
- type Float16
- type Off_t
- type Options
- type Packer
- type Size_t
- type Type
Constants ¶
This section is empty.
Variables ¶
This section is empty.
Functions ¶
func PackWithOrder ¶
Deprecated. Use PackWithOptions.
func SizeofWithOptions ¶
func UnpackWithOptions ¶
Types ¶
type Field ¶
type Field struct { Name string Ptr bool Index int Type Type Array bool Slice bool Len int Order binary.ByteOrder Sizeof []int Sizefrom []int Fields Fields // contains filtered or unexported fields }