Commit 89b5c6c0 authored by Brad Fitzpatrick's avatar Brad Fitzpatrick

encoding/json: faster encoding

The old code was caching per-type struct field info. Instead,
cache type-specific encoding funcs, tailored for that
particular type to avoid unnecessary reflection at runtime.
Once the machine is built once, future encodings of that type
just run the func.

benchmark               old ns/op    new ns/op    delta
BenchmarkCodeEncoder     48424939     36975320  -23.64%

benchmark                old MB/s     new MB/s  speedup
BenchmarkCodeEncoder        40.07        52.48    1.31x

Additionally, the numbers seem stable now at ~52 MB/s, whereas
the numbers for the old code were all over the place: 11 MB/s,
40 MB/s, 13 MB/s, 39 MB/s, etc.  In the benchmark above I compared
against the best I saw the old code do.

R=rsc, adg
CC=gobot, golang-dev, r
https://golang.org/cl/9129044
parent ba6cf63c
...@@ -421,6 +421,45 @@ func TestMarshalNumberZeroVal(t *testing.T) { ...@@ -421,6 +421,45 @@ func TestMarshalNumberZeroVal(t *testing.T) {
} }
} }
func TestMarshalEmbeds(t *testing.T) {
top := &Top{
Level0: 1,
Embed0: Embed0{
Level1b: 2,
Level1c: 3,
},
Embed0a: &Embed0a{
Level1a: 5,
Level1b: 6,
},
Embed0b: &Embed0b{
Level1a: 8,
Level1b: 9,
Level1c: 10,
Level1d: 11,
Level1e: 12,
},
Loop: Loop{
Loop1: 13,
Loop2: 14,
},
Embed0p: Embed0p{
Point: image.Point{X: 15, Y: 16},
},
Embed0q: Embed0q{
Point: Point{Z: 17},
},
}
b, err := Marshal(top)
if err != nil {
t.Fatal(err)
}
want := "{\"Level0\":1,\"Level1b\":2,\"Level1c\":3,\"Level1a\":5,\"LEVEL1B\":6,\"e\":{\"Level1a\":8,\"Level1b\":9,\"Level1c\":10,\"Level1d\":11,\"x\":12},\"Loop1\":13,\"Loop2\":14,\"X\":15,\"Y\":16,\"Z\":17}"
if string(b) != want {
t.Errorf("Wrong marshal result.\n got: %q\nwant: %q", b, want)
}
}
func TestUnmarshal(t *testing.T) { func TestUnmarshal(t *testing.T) {
for i, tt := range unmarshalTests { for i, tt := range unmarshalTests {
var scan scanner var scan scanner
...@@ -436,7 +475,7 @@ func TestUnmarshal(t *testing.T) { ...@@ -436,7 +475,7 @@ func TestUnmarshal(t *testing.T) {
} }
// v = new(right-type) // v = new(right-type)
v := reflect.New(reflect.TypeOf(tt.ptr).Elem()) v := reflect.New(reflect.TypeOf(tt.ptr).Elem())
dec := NewDecoder(bytes.NewBuffer(in)) dec := NewDecoder(bytes.NewReader(in))
if tt.useNumber { if tt.useNumber {
dec.UseNumber() dec.UseNumber()
} }
...@@ -461,7 +500,7 @@ func TestUnmarshal(t *testing.T) { ...@@ -461,7 +500,7 @@ func TestUnmarshal(t *testing.T) {
continue continue
} }
vv := reflect.New(reflect.TypeOf(tt.ptr).Elem()) vv := reflect.New(reflect.TypeOf(tt.ptr).Elem())
dec = NewDecoder(bytes.NewBuffer(enc)) dec = NewDecoder(bytes.NewReader(enc))
if tt.useNumber { if tt.useNumber {
dec.UseNumber() dec.UseNumber()
} }
...@@ -471,6 +510,8 @@ func TestUnmarshal(t *testing.T) { ...@@ -471,6 +510,8 @@ func TestUnmarshal(t *testing.T) {
} }
if !reflect.DeepEqual(v.Elem().Interface(), vv.Elem().Interface()) { if !reflect.DeepEqual(v.Elem().Interface(), vv.Elem().Interface()) {
t.Errorf("#%d: mismatch\nhave: %#+v\nwant: %#+v", i, v.Elem().Interface(), vv.Elem().Interface()) t.Errorf("#%d: mismatch\nhave: %#+v\nwant: %#+v", i, v.Elem().Interface(), vv.Elem().Interface())
t.Errorf(" In: %q", strings.Map(noSpace, string(in)))
t.Errorf("Marshal: %q", strings.Map(noSpace, string(enc)))
continue continue
} }
} }
......
This diff is collapsed.
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment