增加 Golang Map 转 Json 的顺序问题.

Signed-off-by: chen.yang <chen.yang@yuzhen-iot.com>
This commit is contained in:
chen.yang 2021-08-03 09:25:42 +08:00
parent fc2335b757
commit fa33b05e7c
1 changed files with 61 additions and 0 deletions

View File

@ -0,0 +1,61 @@
# [Golang Map 转 Json 的顺序问题](https://blog.csdn.net/xz_studying/article/details/103219478)
我们都知道 map 是无序的,每次取出 key/value 的顺序都可能不一致,但 map 转 json 的顺序是不是也是无序的吗?尽管 json 中的参数顺序大部分情况下对使用没有影响,我们不妨看看源码中怎么处理的。
## 1.结论
* map 转 json 是有序的,按照字符编码升序排列 key。
## 2.溯源
源码位于encoding/json/encode.go中
```go
type mapEncoder struct {
elemEnc encoderFunc
}
func (me mapEncoder) encode(e *encodeState, v reflect.Value, opts encOpts) {
if v.IsNil() {//为nil时返回null
e.WriteString("null")
return
}
e.WriteByte('{')
// Extract and sort the keys.
keys := v.MapKeys()//获取map中的所有keys
sv := make([]reflectWithString, len(keys))
for i, v := range keys {
sv[i].v = v
if err := sv[i].resolve(); err != nil {//处理key尤其是非stringint/uint类型的key转string
e.error(&MarshalerError{v.Type(), err})
}
}
//排序,升序,直接比较字符串
sort.Slice(sv, func(i, j int) bool { return sv[i].s < sv[j].s })
for i, kv := range sv {
if i > 0 {
e.WriteByte(',')
}
e.string(kv.s, opts.escapeHTML)
e.WriteByte(':')
me.elemEnc(e, v.MapIndex(kv.v), opts)
}
e.WriteByte('}')
}
func newMapEncoder(t reflect.Type) encoderFunc {
switch t.Key().Kind() {
case reflect.String,
reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64,
reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
default:
if !t.Key().Implements(textMarshalerType) {
return unsupportedTypeEncoder
}
}
me := mapEncoder{typeEncoder(t.Elem())}
return me.encode
}
```