9.1 JSON

JSON(JavaScript 对象表示,JavaScript Object Notation)作为一种轻量级的数据交换格式1,在今天几乎占据了绝大多数的市场份额。虽然与更紧凑的数据交换格式相比,它的序列化和反序列化性能不足,但是它也提供了良好的可读性与易用性,在不追求机制性能的情况下,JSON 是一种非常好的选择。

9.1.1 设计原理

几乎所有的现代编程语言都会将处理 JSON 的函数直接纳入标准库,Go 语言也不例外,它通过 encoding/json 对外提供标准的 JSON 序列化和反序列化方法,即 encoding/json.Marshalencoding/json.Unmarshal,它们也是包中最常用的两个方法。

json-marshal-and-unmarshal

图 9-1 序列化和反序列化

序列化和反序列化的开销完全不同,JSON 反序列化的开销是序列化开销的好几倍,相信这背后的原因也非常好理解。Go 语言中的 JSON 序列化过程不需要被序列化的对象预先实现任何接口,它会通过反射获取结构体或者数组中的值并以树形的结构递归地进行编码,标准库也会根据 encoding/json.Unmarshal 中传入的值对 JSON 进行解码。

Go 语言 JSON 标准库编码和解码的过程大量地运用了反射这一特性,你会在本节的后半部分看到大量的反射代码,这里就不过多介绍了。我们在这里会简单介绍 JSON 标准库中的接口和标签,这是它为开发者提供的为数不多的影响编解码过程的接口。

接口

JSON 标准库中提供了 encoding/json.Marshalerencoding/json.Unmarshaler 两个接口分别可以影响 JSON 的序列化和反序列化结果:

  1. type Marshaler interface {
  2. MarshalJSON() ([]byte, error)
  3. }
  4. type Unmarshaler interface {
  5. UnmarshalJSON([]byte) error
  6. }

在 JSON 序列化和反序列化的过程中,它们会使用反射判断结构体类型是否实现了上述接口,如果实现了上述接口就会优先使用对应的方法进行编码和解码操作,除了这两个方法之外,Go 语言其实还提供了另外两个用于控制编解码结果的方法,即 encoding.TextMarshalerencoding.TextUnmarshaler

  1. type TextMarshaler interface {
  2. MarshalText() (text []byte, err error)
  3. }
  4. type TextUnmarshaler interface {
  5. UnmarshalText(text []byte) error
  6. }

一旦发现 JSON 相关的序列化方法没有被实现,上述两个方法会作为候选方法被 JSON 标准库调用,参与编解码的过程。总得来说,我们可以在任意类型上实现上述这四个方法自定义最终的结果,后面的两个方法的适用范围更广,但是不会被 JSON 标准库优先调用。

标签

Go 语言的结构体标签也是一个比较有趣的功能,在默认情况下,当我们在序列化和反序列化结构体时,标准库都会认为字段名和 JSON 中的键具有一一对应的关系,然而 Go 语言的字段一般都是驼峰命名法,JSON 中下划线的命名方式相对比较常见,所以使用标签这一特性直接建立键与字段之间的映射关系是一个非常方便的设计。

struct-and-json

图 9-2 结构体与 JSON 的映射

JSON 中的标签由两部分组成,如下所示的 nameage 都是标签名,后面的所有的字符串是标签选项,即 encoding/json.tagOptions,标签名和字段名会建立一一对应的关系,后面的标签选项也会影响编解码的过程:

  1. type Author struct {
  2. Name string `json:"name,omitempty"`
  3. Age int32 `json:"age,string,omitempty"`
  4. }

常见的两个标签是 stringomitempty,前者表示当前的整数或者浮点数是由 JSON 中的字符串表示的,而另一个字段 omitempty 会在字段为零值时,直接在生成的 JSON 中忽略对应的键值对,例如:"age": 0"author": "" 等。标准库会使用 encoding/json.parseTag 函数来解析标签:

  1. func parseTag(tag string) (string, tagOptions) {
  2. if idx := strings.Index(tag, ","); idx != -1 {
  3. return tag[:idx], tagOptions(tag[idx+1:])
  4. }
  5. return tag, tagOptions("")
  6. }

从该方法的实现中,我们能分析出 JSON 标准库中的合法标签是什么形式的 — 标签名和标签选项都以 , 连接,最前面的字符串为标签名,后面的都是标签选项。

9.1.2 序列化

encoding/json.Marshal 是 JSON 标准库中提供的最简单的序列化函数,它会接收一个 interface{} 类型的值作为参数,这也意味着几乎全部的 Go 语言变量都可以被 JSON 标准库序列化,为了提供如此复杂和通用的功能,在静态语言中使用反射是常见的选项,我们来深入了解一下该方法的实现:

  1. func Marshal(v interface{}) ([]byte, error) {
  2. e := newEncodeState()
  3. err := e.marshal(v, encOpts{escapeHTML: true})
  4. if err != nil {
  5. return nil, err
  6. }
  7. buf := append([]byte(nil), e.Bytes()...)
  8. encodeStatePool.Put(e)
  9. return buf, nil
  10. }

上述方法会调用 encoding/json.newEncodeState 从全局的编码状态池中获取 encoding/json.encodeState,随后的序列化过程都会使用这个编码状态,该结构体也会在编码结束后被重新放回池中以便重复利用。

json-marshal-call-stack

图 9-3 序列化调用栈

按照如上所示的复杂调用栈,一系列的序列化方法在最后获取了对象的反射类型并调用了 encoding/json.newTypeEncoder 这个核心的编码方法,该方法会递归地为所有的类型找到对应的编码方法,不过它的执行过程可以分成以下两个步骤:

  1. 获取用户自定义的 encoding/json.Marshaler 或者 encoding.TextMarshaler 编码器;
  2. 获取标准库中为基本类型内置的 JSON 编码器;

在该方法的第一部分,我们会检查当前值的类型是否可以使用用户自定义的编码器,这里有两种不同的判断方法:

  1. func newTypeEncoder(t reflect.Type, allowAddr bool) encoderFunc {
  2. if t.Kind() != reflect.Ptr && allowAddr && reflect.PtrTo(t).Implements(marshalerType) {
  3. return newCondAddrEncoder(addrMarshalerEncoder, newTypeEncoder(t, false))
  4. }
  5. if t.Implements(marshalerType) {
  6. return marshalerEncoder
  7. }
  8. if t.Kind() != reflect.Ptr && allowAddr && reflect.PtrTo(t).Implements(textMarshalerType) {
  9. return newCondAddrEncoder(addrTextMarshalerEncoder, newTypeEncoder(t, false))
  10. }
  11. if t.Implements(textMarshalerType) {
  12. return textMarshalerEncoder
  13. }
  14. ...
  15. }
  1. 如果当前值是值类型、可以取地址并且值类型对应的指针类型实现了 encoding/json.Marshaler 接口,调用 encoding/json.newCondAddrEncoder 获取一个条件编码器,条件编码器会在 encoding/json.addrMarshalerEncoder 失败时重新选择新的编码器;
  2. 如果当前类型实现了 encoding/json.Marshaler 接口,可以直接使用 encoding/json.marshalerEncoder 对该值进行序列化;

在这段代码中,标准库对 encoding.TextMarshaler 的处理也几乎完全相同,只是它会先判断 encoding/json.Marshaler 接口,这也印证了我们在设计原理一节中的结论。

encoding/json.newTypeEncoder 方法随后会根据传入值的反射类型获取对应的编码器,其中包括 boolintfloat 等基本类型编码器等和数组、结构体、切片等复杂类型的编码器:

  1. func newTypeEncoder(t reflect.Type, allowAddr bool) encoderFunc {
  2. ...
  3. switch t.Kind() {
  4. case reflect.Bool:
  5. return boolEncoder
  6. case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64:
  7. return intEncoder
  8. case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr:
  9. return uintEncoder
  10. case reflect.Float32:
  11. return float32Encoder
  12. case reflect.Float64:
  13. return float64Encoder
  14. case reflect.String:
  15. return stringEncoder
  16. case reflect.Interface:
  17. return interfaceEncoder
  18. case reflect.Struct:
  19. return newStructEncoder(t)
  20. case reflect.Map:
  21. return newMapEncoder(t)
  22. case reflect.Slice:
  23. return newSliceEncoder(t)
  24. case reflect.Array:
  25. return newArrayEncoder(t)
  26. case reflect.Ptr:
  27. return newPtrEncoder(t)
  28. default:
  29. return unsupportedTypeEncoder
  30. }
  31. }

我们在这里就不一一介绍全部的内置类型编码器了,只挑选其中的几个帮助各位读者了解整体的设计。首先我们来看布尔值的 JSON 编码器,它的实现很简单,甚至没有太多值得介绍的地方:

  1. func boolEncoder(e *encodeState, v reflect.Value, opts encOpts) {
  2. if opts.quoted {
  3. e.WriteByte('"')
  4. }
  5. if v.Bool() {
  6. e.WriteString("true")
  7. } else {
  8. e.WriteString("false")
  9. }
  10. if opts.quoted {
  11. e.WriteByte('"')
  12. }
  13. }

它会根据当前值向编码状态中写入不同的字符串,也就是 true 或者 false,除此之外还会根据编码配置决定是否要在布尔值周围写入双引号 ",而其他的基本类型编码器也都大同小异。

复杂类型的编码器有着相对复杂的控制结构,我们在这里以结构体的编码器 encoding/json.structEncoder 为例介绍它们的原理,encoding/json.newStructEncoder 会为当前结构体的所有字段调用 encoding/json.typeEncoder 获取类型编码器并返回 encoding/json.structEncoder.encode 方法:

  1. func newStructEncoder(t reflect.Type) encoderFunc {
  2. se := structEncoder{fields: cachedTypeFields(t)}
  3. return se.encode
  4. }

encoding/json.structEncoder.encode 的实现我们能看出结构体序列的结果,该方法会遍历结构体中的全部字段,在写入了字段名后,它会调用字段对应类型的编码方法将该字段对应的 JSON 写入缓冲区:

  1. func (se structEncoder) encode(e *encodeState, v reflect.Value, opts encOpts) {
  2. next := byte('{')
  3. FieldLoop:
  4. for i := range se.fields.list {
  5. f := &se.fields.list[i]
  6. fv := v
  7. for _, i := range f.index {
  8. if fv.Kind() == reflect.Ptr {
  9. if fv.IsNil() {
  10. continue FieldLoop
  11. }
  12. fv = fv.Elem()
  13. }
  14. fv = fv.Field(i)
  15. }
  16. if f.omitEmpty && isEmptyValue(fv) {
  17. continue
  18. }
  19. e.WriteByte(next)
  20. next = ','
  21. e.WriteString(f.nameNonEsc)
  22. opts.quoted = f.quoted
  23. f.encoder(e, fv, opts)
  24. }
  25. if next == '{' {
  26. e.WriteString("{}")
  27. } else {
  28. e.WriteByte('}')
  29. }
  30. }

数组以及指针等编码器的实现原理与该方法也没有太多的区别,它们都会使用类似的策略递归地调用持有字段的编码方法,这也就能形成一个如下图所示的树形结构:

struct-encoder

图 9-4 序列化与树形结构体

树形结构的所有叶节点都是基础类型编码器或者开发者自定义的编码器,得到了整棵树的编码器之后会调用 encoding/json.encodeState.reflectValue 从根节点依次调用整棵树的序列化函数,整个 JSON 序列化的过程其实是查找类型和子类型的编码方法并调用的过程,它利用了大量反射的特性做到了足够的通用。

9.1.3 反序列化

标准库会使用 encoding/json.Unmarshal 函数处理 JSON 的反序列化,与执行过程确定的序列化相比,反序列化的过程比较像一个逐渐探索的过程,所以会复杂很多,开销也会高出几倍。因为 Go 语言的表达能力比较有限,反序列化的使用相对比较繁琐,需要传入一个变量帮助标准库进行反序列化:

  1. func Unmarshal(data []byte, v interface{}) error {
  2. var d decodeState
  3. err := checkValid(data, &d.scan)
  4. if err != nil {
  5. return err
  6. }
  7. d.init(data)
  8. return d.unmarshal(v)
  9. }

在真正执行反序列化之前,我们会先调用 encoding/json.checkValid 验证传入 JSON 的合法性保证在反序列化的过程中不会遇到语法错误的问题,在通过合法性的验证之后,标准库就会初始化数据并调用 encoding/json.decodeState.unmarshal 开始反序列化了:

  1. func (d *decodeState) unmarshal(v interface{}) error {
  2. rv := reflect.ValueOf(v)
  3. if rv.Kind() != reflect.Ptr || rv.IsNil() {
  4. return &InvalidUnmarshalError{reflect.TypeOf(v)}
  5. }
  6. d.scan.reset()
  7. d.scanWhile(scanSkipSpace)
  8. err := d.value(rv)
  9. if err != nil {
  10. return d.addErrorContext(err)
  11. }
  12. return d.savedError
  13. }

如果传入的值不是指针或者是空指针,当前方法就会返回我们经常会见到的错误 encoding/json.InvalidUnmarshalError,使用格式化输出可以将该错误转换成 json: Unmarshal(non-pointer xxx)。该方法调用的 encoding/json.decodeState.value 是所有反序列化过程的执行入口:

  1. func (d *decodeState) value(v reflect.Value) error {
  2. switch d.opcode {
  3. default:
  4. panic(phasePanicMsg)
  5. case scanBeginArray:
  6. ...
  7. case scanBeginLiteral:
  8. ...
  9. case scanBeginObject:
  10. if v.IsValid() {
  11. if err := d.object(v); err != nil {
  12. return err
  13. }
  14. } else {
  15. d.skip()
  16. }
  17. d.scanNext()
  18. }
  19. return nil
  20. }

该方法作为最顶层的反序列化方法可以接收三种不同类型的值,也就是数组、字面量和对象,这三种类型都可以作为 JSON 的顶层对象,我们首先来了解一下标准库是如何解析 JSON 中对象的,该过程会使用 encoding/json.decodeState.object 函数进行反序列化,它会先调用 encoding/json.indirect 函数查找当前类型对应的非指针类型:

  1. func (d *decodeState) object(v reflect.Value) error {
  2. u, ut, pv := indirect(v, false)
  3. if u != nil {
  4. start := d.readIndex()
  5. d.skip()
  6. return u.UnmarshalJSON(d.data[start:d.off])
  7. }
  8. ...
  9. }

在调用 encoding/json.indirect 的过程中,如果当前值的类型是 **Type,那么它会依次检查形如 **Type*TypeType 类型是否实现了 encoding/json.Unmarshal 或者 encoding.TextUnmarshaler 接口;如果实现了该接口,标准库会直接调用 UnmarshalJSON 方法使用开发者定义的方法完成反序列化。

在其他情况下,我们仍然会回到默认的逻辑中处理对象中的键值对,如下所示的代码会调用 encoding/json.decodeState.rescanLiteral 方法扫描 JSON 中的键并在结构体中找到对应字段的反射值,接下来继续扫描符号 : 并调用 encoding/json.decodeState.value 解析对应的值:

  1. func (d *decodeState) object(v reflect.Value) error {
  2. ...
  3. v = pv
  4. t := v.Type()
  5. fields = cachedTypeFields(t)
  6. for {
  7. start := d.readIndex()
  8. d.rescanLiteral()
  9. item := d.data[start:d.readIndex()]
  10. key, _ := d.unquoteBytes(item)
  11. var subv reflect.Value
  12. var f *field
  13. if i, ok := fields.nameIndex[string(key)]; ok {
  14. f = &fields.list[i]
  15. }
  16. if f != nil {
  17. subv = v
  18. for _, i := range f.index {
  19. subv = subv.Field(i)
  20. }
  21. }
  22. if d.opcode != scanObjectKey {
  23. panic(phasePanicMsg)
  24. }
  25. d.scanWhile(scanSkipSpace)
  26. if err := d.value(subv); err != nil {
  27. return err
  28. }
  29. if d.opcode == scanEndObject {
  30. break
  31. }
  32. }
  33. return nil
  34. }

当上述方法调用 encoding/json.decodeState.value 时,该方法会重新判断键对应的值是否是对象、数组或者字面量,因为数组和对象都是集合类型,所以该方法会递归地进行扫描,在这里也就不再继续介绍这些集合类型的解析过程了,我们来简单分析一下字面量是如何被处理的:

  1. func (d *decodeState) value(v reflect.Value) error {
  2. switch d.opcode {
  3. default:
  4. panic(phasePanicMsg)
  5. case scanBeginArray:
  6. ...
  7. case scanBeginObject:
  8. ...
  9. case scanBeginLiteral:
  10. start := d.readIndex()
  11. d.rescanLiteral()
  12. if v.IsValid() {
  13. if err := d.literalStore(d.data[start:d.readIndex()], v, false); err != nil {
  14. return err
  15. }
  16. }
  17. }
  18. return nil
  19. }

字面量的扫描会通过 encoding/json.decodeState.rescanLiteral,该方法会依次扫描缓冲区中的字符并根据字符的不同对字符串进行切片,整个过程有点像编译器的词法分析:

  1. func (d *decodeState) rescanLiteral() {
  2. data, i := d.data, d.off
  3. Switch:
  4. switch data[i-1] {
  5. case '"': // string
  6. ...
  7. case '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '-': // number
  8. ...
  9. case 't': // true
  10. i += len("rue")
  11. case 'f': // false
  12. i += len("alse")
  13. case 'n': // null
  14. i += len("ull")
  15. }
  16. if i < len(data) {
  17. d.opcode = stateEndValue(&d.scan, data[i])
  18. } else {
  19. d.opcode = scanEnd
  20. }
  21. d.off = i + 1
  22. }

因为 JSON 中的字面量其实也只包含字符串、数字、布尔值和空值几种,所以该方法的实现也不会特别复杂,当该方法扫描了对应的字面量之后,就会调用 encoding/json.decodeState.literalStore 字面量存储到反射类型变量所在的地址中,在这个过程中会调用反射的 reflect.Value.SetIntreflect.Value.SetFloatreflect.Value.SetBool 等方法。

9.1.4 小结

JSON 本身就是一种树形的数据结构,无论是序列化还是反序列化,都会遵循自顶向下的编码和解码过程,使用递归的方式处理 JSON 对象。作为标准库的 JSON 提供的接口非常简洁,虽然它的性能一直被开发者所诟病,但是作为框架它提供了很好的通用性,通过分析 JSON 库的实现,我们也可以从中学习到使用反射的各种方法。


  1. Introducing JSON https://www.json.org/json-en.html ↩︎

wechat-account-qrcode

本作品采用知识共享署名-非商业性使用-禁止演绎 4.0 国际许可协议进行许可。