簡介
easyjson是什麼呢? 根據官網介紹,easyjson是提供高效快速且易用的結構體structs<-->json轉換包。easyjson並沒有使用反射方式實現,所以效能比其他的json包該4-5倍,比golang 內建的json包快2-3倍。 easyjson目標是維持產生去代碼簡單,以致於它可以輕鬆地進行最佳化或固定。
安裝
go get -u github.com/mailru/easyjson/go install github.com/mailru/easyjson/easyjsonorgo build -o easyjson github.com/mailru/easyjson/easyjson
驗證是否安裝成功。
$ easyjsonUsage of D:\Code\go\bin\easyjson.exe: -all generate marshaler/unmarshalers for all structs in a file -build_tags string build tags to add to generated file -leave_temps do not delete temporary files -lower_camel_case use lowerCamelCase names instead of CamelCase by default -no_std_marshalers don't generate MarshalJSON/UnmarshalJSON funcs -noformat do not run 'gofmt -w' on output file -omit_empty omit empty fields by default string specify the filename of the output -pkg process the whole package instead of just the given file -snake_case use snake_case names instead of CamelCase by default -stubs only generate stubs for marshaler/unmarshaler funcs
其中有幾個選項需要注意:
-lower_camel_case:將結構體欄位field首字母改為小寫。如Name=>name。 -build_tags string:將指定的string產生到產生的go檔案頭部。 -no_std_marshalers:不為結構體產生MarshalJSON/UnmarshalJSON函數。 -omit_empty:沒有賦值的field可以不產生到json,否則field為該欄位類型的預設值。-output_filename:定義產生的檔案名稱。-pkg:對包內指定有`//easyjson:json`結構體產生對應的easyjson配置。-snke_case:可以底線的field如`Name_Student`改為`name_student`。
使用
記得在需要使用easyjson
的結構體上加上//easyjson:json
。 如下:
//easyjson:jsontype School struct { Name string `json:"name"` Addr string `json:"addr"`}//easyjson:jsontype Student struct { Id int `json:"id"` Name string `json:"s_name"` School School `json:"s_chool"` Birthday time.Time `json:"birthday"`}
在結構體包下執行
easyjson -all student.go
此時在該目錄下出現一個新的檔案。
// Code generated by easyjson for marshaling/unmarshaling. DO NOT EDIT.package easyjsonimport ( json "encoding/json" easyjson "github.com/mailru/easyjson" jlexer "github.com/mailru/easyjson/jlexer" jwriter "github.com/mailru/easyjson/jwriter")// suppress unused package warningvar ( _ *json.RawMessage _ *jlexer.Lexer _ *jwriter.Writer _ easyjson.Marshaler)func easyjsonB83d7b77DecodeStudygoEasyjson(in *jlexer.Lexer, out *Student) { isTopLevel := in.IsStart() if in.IsNull() { if isTopLevel { in.Consumed() } in.Skip() return } in.Delim('{') for !in.IsDelim('}') { key := in.UnsafeString() in.WantColon() if in.IsNull() { in.Skip() in.WantComma() continue } switch key { case "id": out.Id = int(in.Int()) case "s_name": out.Name = string(in.String()) case "s_chool": easyjsonB83d7b77DecodeStudygoEasyjson1(in, &out.School) case "birthday": if data := in.Raw(); in.Ok() { in.AddError((out.Birthday).UnmarshalJSON(data)) } default: in.SkipRecursive() } in.WantComma() } in.Delim('}') if isTopLevel { in.Consumed() }}func easyjsonB83d7b77EncodeStudygoEasyjson(out *jwriter.Writer, in Student) { out.RawByte('{') first := true _ = first if !first { out.RawByte(',') } first = false out.RawString("\"id\":") out.Int(int(in.Id)) if !first { out.RawByte(',') } first = false out.RawString("\"s_name\":") out.String(string(in.Name)) if !first { out.RawByte(',') } first = false out.RawString("\"s_chool\":") easyjsonB83d7b77EncodeStudygoEasyjson1(out, in.School) if !first { out.RawByte(',') } first = false out.RawString("\"birthday\":") out.Raw((in.Birthday).MarshalJSON()) out.RawByte('}')}// MarshalJSON supports json.Marshaler interfacefunc (v Student) MarshalJSON() ([]byte, error) { w := jwriter.Writer{} easyjsonB83d7b77EncodeStudygoEasyjson(&w, v) return w.Buffer.BuildBytes(), w.Error}// MarshalEasyJSON supports easyjson.Marshaler interfacefunc (v Student) MarshalEasyJSON(w *jwriter.Writer) { easyjsonB83d7b77EncodeStudygoEasyjson(w, v)}// UnmarshalJSON supports json.Unmarshaler interfacefunc (v *Student) UnmarshalJSON(data []byte) error { r := jlexer.Lexer{Data: data} easyjsonB83d7b77DecodeStudygoEasyjson(&r, v) return r.Error()}// UnmarshalEasyJSON supports easyjson.Unmarshaler interfacefunc (v *Student) UnmarshalEasyJSON(l *jlexer.Lexer) { easyjsonB83d7b77DecodeStudygoEasyjson(l, v)}func easyjsonB83d7b77DecodeStudygoEasyjson1(in *jlexer.Lexer, out *School) { isTopLevel := in.IsStart() if in.IsNull() { if isTopLevel { in.Consumed() } in.Skip() return } in.Delim('{') for !in.IsDelim('}') { key := in.UnsafeString() in.WantColon() if in.IsNull() { in.Skip() in.WantComma() continue } switch key { case "name": out.Name = string(in.String()) case "addr": out.Addr = string(in.String()) default: in.SkipRecursive() } in.WantComma() } in.Delim('}') if isTopLevel { in.Consumed() }}func easyjsonB83d7b77EncodeStudygoEasyjson1(out *jwriter.Writer, in School) { out.RawByte('{') first := true _ = first if !first { out.RawByte(',') } first = false out.RawString("\"name\":") out.String(string(in.Name)) if !first { out.RawByte(',') } first = false out.RawString("\"addr\":") out.String(string(in.Addr)) out.RawByte('}')}
現在可以寫一個測試類別啦。
package mainimport ( "studygo/easyjson" "time" "fmt")func main(){ s:=easyjson.Student{ Id: 11, Name:"qq", School:easyjson.School{ Name:"CUMT", Addr:"xz", }, Birthday:time.Now(), } bt,err:=s.MarshalJSON() fmt.Println(string(bt),err) json:=`{"id":11,"s_name":"qq","s_chool":{"name":"CUMT","addr":"xz"},"birthday":"2017-08-04T20:58:07.9894603+08:00"}` ss:=easyjson.Student{} ss.UnmarshalJSON([]byte(json)) fmt.Println(ss)}
運行結果:
{"id":11,"s_name":"qq","s_chool":{"name":"CUMT","addr":"xz"},"birthday":"2017-08-04T20:58:07.9894603+08:00"} <nil>{121 {CwwwwwwwUMT xzwwwww} 2017-08-04 20:52:03.4066002 +0800 CST}