Go语言的JSON库通过反射在运行时获取值的类型和字段信息,自动处理结构体、切片、映射等数据的序列化与反序列化,无需实现特定接口;序列化时利用reflect.Value和reflect.Type遍历可导出字段,读取json标签确定键名,并根据字段类型递归生成JSON,同时处理nil指针和零值情况。

Go语言的JSON库(encoding/json)在序列化和反序列化过程中广泛使用反射(reflect),以便在运行时动态地读取和操作任意类型的值。它不需要类型实现特定接口,就能将结构体、切片、映射等数据结构转换为JSON字符串,或从JSON还原为Go值。
反射如何参与序列化(Go值 → JSON)
当调用 json.Marshal 时,Go会通过反射获取输入值的类型和字段信息:
- 使用 reflect.Value 获取值的运行时值,reflect.Type 获取其类型元数据。
- 遍历结构体字段时,检查每个字段是否可导出(首字母大写),只有可导出字段才会被处理。
- 读取结构体字段上的 json标签(如 json:"name"),决定该字段在JSON中的键名。
- 根据字段类型(字符串、数字、布尔、切片、嵌套结构体等)递归生成对应的JSON结构。
- 如果字段值为 nil 指针或零值且有
type User struct {
Name string `json:"name"`
Age int `json:"age,omitempty"`
}反射会识别 Name 字段对应 JSON 的 name,而 Age 在为0时不会出现在输出中。
立即学习“go语言免费学习笔记(深入)”;
反射如何参与反序列化(JSON → Go值)
在调用 json.Unmarshal 时,目标参数必须是指针,以便修改其内容。反射在此过程中起核心作用:
- 通过指针获取指向的值的 reflect.Value,然后设置其字段。
- 解析JSON对象的键,匹配结构体字段:先尝试通过 json标签 匹配,再尝试直接字段名匹配(大小写敏感)。
- 对每个匹配的字段,使用反射设置其值。例如将JSON字符串赋给字符串字段,数字赋给int字段。
- 如果结构体字段是结构体、指针、切片或映射,反射会递归分配内存并填充内容。
- 未匹配的JSON字段默认被忽略,除非结构体中有 json:"-" 或使用了额外字段(如 map[string]interface{})。
例如,解析 {"name": "Alice", "age": 30} 到 User 指针时,反射会定位到 Name 和 Age 字段并赋值。
性能与限制
反射虽然灵活,但有一定开销:
- 类型检查和字段查找发生在运行时,比静态编码慢。
- 无法处理不支持JSON表示的类型,如 chan、func、复杂嵌套指针等。
- 私有字段(小写开头)无法被反射写入,因此不会被反序列化。
为提升性能,可使用 json tags 明确控制映射,或借助代码生成工具(如 ffjson、easyjson)生成无反射的编解码函数。
基本上就这些。Go的JSON库靠反射实现了通用性,让开发者能用极少代码完成数据交换,虽牺牲一点性能,但换来了简洁和易用。










