1
0
mirror of https://github.com/json-iterator/go.git synced 2025-04-01 21:24:21 +02:00

#63 keep struct field order

This commit is contained in:
Tao Wen 2017-06-23 08:21:02 +08:00
parent d7ea1acd3f
commit 8f8e16b4c2
6 changed files with 72 additions and 47 deletions

View File

@ -144,7 +144,6 @@ func (encoder *htmlEscapedStringEncoder) IsEmpty(ptr unsafe.Pointer) bool {
} }
func (cfg *frozenConfig) escapeHtml() { func (cfg *frozenConfig) escapeHtml() {
// for better performance
cfg.addEncoderToCache(reflect.TypeOf((*string)(nil)).Elem(), &htmlEscapedStringEncoder{}) cfg.addEncoderToCache(reflect.TypeOf((*string)(nil)).Elem(), &htmlEscapedStringEncoder{})
} }
@ -192,14 +191,14 @@ func (cfg *frozenConfig) getEncoderFromCache(cacheKey reflect.Type) ValEncoder {
func (cfg *frozenConfig) cleanDecoders() { func (cfg *frozenConfig) cleanDecoders() {
typeDecoders = map[string]ValDecoder{} typeDecoders = map[string]ValDecoder{}
fieldDecoders = map[string]ValDecoder{} fieldDecoders = map[string]ValDecoder{}
atomic.StorePointer(&cfg.decoderCache, unsafe.Pointer(&map[string]ValDecoder{})) *cfg = *cfg.configBeforeFrozen.Froze()
} }
// cleanEncoders cleans encoders registered or cached // cleanEncoders cleans encoders registered or cached
func (cfg *frozenConfig) cleanEncoders() { func (cfg *frozenConfig) cleanEncoders() {
typeEncoders = map[string]ValEncoder{} typeEncoders = map[string]ValEncoder{}
fieldEncoders = map[string]ValEncoder{} fieldEncoders = map[string]ValEncoder{}
atomic.StorePointer(&cfg.encoderCache, unsafe.Pointer(&map[string]ValEncoder{})) *cfg = *cfg.configBeforeFrozen.Froze()
} }
func (cfg *frozenConfig) MarshalToString(v interface{}) (string, error) { func (cfg *frozenConfig) MarshalToString(v interface{}) (string, error) {

View File

@ -192,7 +192,8 @@ func _getTypeEncoderFromExtension(typ reflect.Type) ValEncoder {
} }
func describeStruct(cfg *frozenConfig, typ reflect.Type) (*StructDescriptor, error) { func describeStruct(cfg *frozenConfig, typ reflect.Type) (*StructDescriptor, error) {
anonymousBindings := []*Binding{} headAnonymousBindings := []*Binding{}
tailAnonymousBindings := []*Binding{}
bindings := []*Binding{} bindings := []*Binding{}
for i := 0; i < typ.NumField(); i++ { for i := 0; i < typ.NumField(); i++ {
field := typ.Field(i) field := typ.Field(i)
@ -205,7 +206,11 @@ func describeStruct(cfg *frozenConfig, typ reflect.Type) (*StructDescriptor, err
for _, binding := range structDescriptor.Fields { for _, binding := range structDescriptor.Fields {
binding.Encoder = &structFieldEncoder{&field, binding.Encoder, false} binding.Encoder = &structFieldEncoder{&field, binding.Encoder, false}
binding.Decoder = &structFieldDecoder{&field, binding.Decoder} binding.Decoder = &structFieldDecoder{&field, binding.Decoder}
anonymousBindings = append(anonymousBindings, binding) if field.Offset == 0 {
headAnonymousBindings = append(headAnonymousBindings, binding)
} else {
tailAnonymousBindings = append(tailAnonymousBindings, binding)
}
} }
} else if field.Type.Kind() == reflect.Ptr && field.Type.Elem().Kind() == reflect.Struct { } else if field.Type.Kind() == reflect.Ptr && field.Type.Elem().Kind() == reflect.Struct {
structDescriptor, err := describeStruct(cfg, field.Type.Elem()) structDescriptor, err := describeStruct(cfg, field.Type.Elem())
@ -217,7 +222,11 @@ func describeStruct(cfg *frozenConfig, typ reflect.Type) (*StructDescriptor, err
binding.Encoder = &structFieldEncoder{&field, binding.Encoder, false} binding.Encoder = &structFieldEncoder{&field, binding.Encoder, false}
binding.Decoder = &optionalDecoder{field.Type, binding.Decoder} binding.Decoder = &optionalDecoder{field.Type, binding.Decoder}
binding.Decoder = &structFieldDecoder{&field, binding.Decoder} binding.Decoder = &structFieldDecoder{&field, binding.Decoder}
anonymousBindings = append(anonymousBindings, binding) if field.Offset == 0 {
headAnonymousBindings = append(headAnonymousBindings, binding)
} else {
tailAnonymousBindings = append(tailAnonymousBindings, binding)
}
} }
} }
} else { } else {
@ -276,7 +285,8 @@ func describeStruct(cfg *frozenConfig, typ reflect.Type) (*StructDescriptor, err
binding.Encoder = &structFieldEncoder{binding.Field, binding.Encoder, shouldOmitEmpty} binding.Encoder = &structFieldEncoder{binding.Field, binding.Encoder, shouldOmitEmpty}
} }
// insert anonymous bindings to the head // insert anonymous bindings to the head
structDescriptor.Fields = append(anonymousBindings, structDescriptor.Fields...) structDescriptor.Fields = append(headAnonymousBindings, structDescriptor.Fields...)
structDescriptor.Fields = append(structDescriptor.Fields, tailAnonymousBindings...)
return structDescriptor, nil return structDescriptor, nil
} }

View File

@ -8,20 +8,35 @@ import (
) )
func encoderOfStruct(cfg *frozenConfig, typ reflect.Type) (ValEncoder, error) { func encoderOfStruct(cfg *frozenConfig, typ reflect.Type) (ValEncoder, error) {
fields := map[string]*structFieldEncoder{} fieldsByToName := map[string]int{}
orderedFields := []*structFieldTo{}
structDescriptor, err := describeStruct(cfg, typ) structDescriptor, err := describeStruct(cfg, typ)
if err != nil { if err != nil {
return nil, err return nil, err
} }
for _, binding := range structDescriptor.Fields { for index, binding := range structDescriptor.Fields {
for _, toName := range binding.ToNames { for _, toName := range binding.ToNames {
fields[toName] = binding.Encoder.(*structFieldEncoder) oldIndex, found := fieldsByToName[toName]
if found {
orderedFields[oldIndex] = nil // replaced by the later one
}
fieldsByToName[toName] = index
orderedFields = append(orderedFields, &structFieldTo{
binding.Encoder.(*structFieldEncoder),
toName,
})
} }
} }
if len(fields) == 0 { if len(orderedFields) == 0 {
return &emptyStructEncoder{}, nil return &emptyStructEncoder{}, nil
} }
return &structEncoder{fields}, nil finalOrderedFields := []structFieldTo{}
for _, structFieldTo := range orderedFields {
if structFieldTo != nil {
finalOrderedFields = append(finalOrderedFields, *structFieldTo)
}
}
return &structEncoder{finalOrderedFields}, nil
} }
func decoderOfStruct(cfg *frozenConfig, typ reflect.Type) (ValDecoder, error) { func decoderOfStruct(cfg *frozenConfig, typ reflect.Type) (ValDecoder, error) {
@ -977,21 +992,26 @@ func (encoder *structFieldEncoder) IsEmpty(ptr unsafe.Pointer) bool {
} }
type structEncoder struct { type structEncoder struct {
fields map[string]*structFieldEncoder fields []structFieldTo
}
type structFieldTo struct {
encoder *structFieldEncoder
toName string
} }
func (encoder *structEncoder) Encode(ptr unsafe.Pointer, stream *Stream) { func (encoder *structEncoder) Encode(ptr unsafe.Pointer, stream *Stream) {
stream.WriteObjectStart() stream.WriteObjectStart()
isNotFirst := false isNotFirst := false
for fieldName, field := range encoder.fields { for _, field := range encoder.fields {
if field.omitempty && field.IsEmpty(ptr) { if field.encoder.omitempty && field.encoder.IsEmpty(ptr) {
continue continue
} }
if isNotFirst { if isNotFirst {
stream.WriteMore() stream.WriteMore()
} }
stream.WriteObjectField(fieldName) stream.WriteObjectField(field.toName)
field.Encode(ptr, stream) field.encoder.Encode(ptr, stream)
isNotFirst = true isNotFirst = true
} }
stream.WriteObjectEnd() stream.WriteObjectEnd()
@ -1001,23 +1021,23 @@ func (encoder *structEncoder) EncodeInterface(val interface{}, stream *Stream) {
var encoderToUse ValEncoder var encoderToUse ValEncoder
encoderToUse = encoder encoderToUse = encoder
if len(encoder.fields) == 1 { if len(encoder.fields) == 1 {
var firstField *structFieldEncoder firstFieldName := encoder.fields[0].toName
var firstFieldName string firstField := encoder.fields[0].encoder
for fieldName, field := range encoder.fields {
firstFieldName = fieldName
firstField = field
}
firstEncoder := firstField.fieldEncoder firstEncoder := firstField.fieldEncoder
firstEncoderName := reflect.TypeOf(firstEncoder).String() firstEncoderName := reflect.TypeOf(firstEncoder).String()
// interface{} has inline optimization for this case // interface{} has inline optimization for this case
if firstEncoderName == "*jsoniter.optionalEncoder" { if firstEncoderName == "*jsoniter.optionalEncoder" {
encoderToUse = &structEncoder{ encoderToUse = &structEncoder{
fields: map[string]*structFieldEncoder{ fields: []structFieldTo{
firstFieldName: { {
toName: firstFieldName,
encoder: &structFieldEncoder{
field: firstField.field, field: firstField.field,
fieldEncoder: firstEncoder.(*optionalEncoder).valueEncoder, fieldEncoder: firstEncoder.(*optionalEncoder).valueEncoder,
omitempty: firstField.omitempty, omitempty: firstField.omitempty,
}}, },
},
},
} }
} }
} }
@ -1026,7 +1046,7 @@ func (encoder *structEncoder) EncodeInterface(val interface{}, stream *Stream) {
func (encoder *structEncoder) IsEmpty(ptr unsafe.Pointer) bool { func (encoder *structEncoder) IsEmpty(ptr unsafe.Pointer) bool {
for _, field := range encoder.fields { for _, field := range encoder.fields {
if !field.IsEmpty(ptr) { if !field.encoder.IsEmpty(ptr) {
return false return false
} }
} }

View File

@ -252,7 +252,7 @@ func (stream *Stream) WriteStringWithHtmlEscaped(s string) {
func writeStringSlowPathWithHtmlEscaped(stream *Stream, i int, s string, valLen int) { func writeStringSlowPathWithHtmlEscaped(stream *Stream, i int, s string, valLen int) {
start := i start := i
// for the remaining parts, we process them char by char // for the remaining parts, we process them char by char
for ; i < valLen; { for i < valLen {
if b := s[i]; b < utf8.RuneSelf { if b := s[i]; b < utf8.RuneSelf {
if htmlSafeSet[b] { if htmlSafeSet[b] {
i++ i++
@ -351,7 +351,7 @@ func (stream *Stream) WriteString(s string) {
func writeStringSlowPath(stream *Stream, i int, s string, valLen int) { func writeStringSlowPath(stream *Stream, i int, s string, valLen int) {
start := i start := i
// for the remaining parts, we process them char by char // for the remaining parts, we process them char by char
for ; i < valLen; { for i < valLen {
if b := s[i]; b < utf8.RuneSelf { if b := s[i]; b < utf8.RuneSelf {
if safeSet[b] { if safeSet[b] {
i++ i++

View File

@ -349,10 +349,7 @@ func Test_multiple_level_anonymous_struct(t *testing.T) {
obj := Level3{Level2{Level1{"1"}, "2"}, "3"} obj := Level3{Level2{Level1{"1"}, "2"}, "3"}
output, err := MarshalToString(obj) output, err := MarshalToString(obj)
should.Nil(err) should.Nil(err)
fmt.Println(output) should.Equal(`{"Field1":"1","Field2":"2","Field3":"3"}`, output)
should.Contains(output, `"Field1":"1"`)
should.Contains(output, `"Field2":"2"`)
should.Contains(output, `"Field3":"3"`)
} }
func Test_multiple_level_anonymous_struct_with_ptr(t *testing.T) { func Test_multiple_level_anonymous_struct_with_ptr(t *testing.T) {
@ -421,9 +418,7 @@ func Test_embed_at_last(t *testing.T) {
s := Struct{Field: "field", FieldType: "field_type", Base: Base{"type"}} s := Struct{Field: "field", FieldType: "field_type", Base: Base{"type"}}
output, err := MarshalToString(s) output, err := MarshalToString(s)
should.Nil(err) should.Nil(err)
should.Contains(output, `"type":"type"`) should.Equal(`{"field":"field","field_type":"field_type","type":"type"}`, output)
should.Contains(output, `"field":"field"`)
should.Contains(output, `"field_type":"field_type"`)
} }
func Test_decode_nested(t *testing.T) { func Test_decode_nested(t *testing.T) {

View File

@ -138,28 +138,29 @@ func Test_string_encode_with_std_without_html_escape(t *testing.T) {
func Test_unicode(t *testing.T) { func Test_unicode(t *testing.T) {
should := require.New(t) should := require.New(t)
output , _ := MarshalToString(map[string]interface{}{"a": "数字山谷"}) output, _ := MarshalToString(map[string]interface{}{"a": "数字山谷"})
should.Equal(`{"a":"数字山谷"}`, output) should.Equal(`{"a":"数字山谷"}`, output)
output , _ = Config{EscapeHtml: false}.Froze().MarshalToString(map[string]interface{}{"a": "数字山谷"}) output, _ = Config{EscapeHtml: false}.Froze().MarshalToString(map[string]interface{}{"a": "数字山谷"})
should.Equal(`{"a":"数字山谷"}`, output) should.Equal(`{"a":"数字山谷"}`, output)
} }
func Test_unicode_and_escape(t *testing.T) { func Test_unicode_and_escape(t *testing.T) {
should := require.New(t) should := require.New(t)
output , err := MarshalToString(`"数字山谷"`) output, err := MarshalToString(`"数字山谷"`)
should.Nil(err) should.Nil(err)
should.Equal(`"\"数字山谷\""`, output) should.Equal(`"\"数字山谷\""`, output)
output , err = ConfigFastest.MarshalToString(`"数字山谷"`) output, err = ConfigFastest.MarshalToString(`"数字山谷"`)
should.Nil(err) should.Nil(err)
should.Equal(`"\"数字山谷\""`, output) should.Equal(`"\"数字山谷\""`, output)
} }
func Test_unsafe_unicode(t *testing.T) { func Test_unsafe_unicode(t *testing.T) {
ConfigDefault.cleanEncoders()
should := require.New(t) should := require.New(t)
output , err := MarshalToString("he\u2029\u2028he") output, err := ConfigDefault.MarshalToString("he\u2029\u2028he")
should.Nil(err) should.Nil(err)
should.Equal(`"he\u2029\u2028he"`, output) should.Equal(`"he\u2029\u2028he"`, output)
output , err = ConfigFastest.MarshalToString("he\u2029\u2028he") output, err = ConfigFastest.MarshalToString("he\u2029\u2028he")
should.Nil(err) should.Nil(err)
should.Equal("\"he\u2029\u2028he\"", output) should.Equal("\"he\u2029\u2028he\"", output)
} }