Golang反射读取JSON标签_Go语言序列化原理解析

13次阅读

应使用 reflect.StructField.Tag.Get(“json”) 解析标签,它正确处理空格、引号和转义;json:”-” 返回空字符串,json:”name,omitempty” 需 trimSuffix 剥离 omitempty;未声明标签时也返回空字符串;json 包对类型缓存字段信息,避免重复反射;自定义 MarshalJSON 应通过 type Alias 防递归并复用标准逻辑。

Golang反射读取JSON标签_Go语言序列化原理解析

反射怎么拿到 struct 字段的 json 标签值

goreflect.StructField.Tag 是一个字符串,但不能直接用 == 判断或截取。必须用 reflect.StructTag.Get("json") 解析——它内部按空格分隔、识别引号、处理转义,比手动 strings.Split 可靠得多。

常见错误是直接访问 field.Tag 字符串并尝试正则匹配,结果在字段有 json:"name,omitempty"json:"-" 时出错。

  • json:"-" 表示忽略该字段,Get("json") 返回空字符串
  • json:"name,omitempty" 会返回 "name,omitempty",需进一步用 strings.TrimSuffix 剥离 ,omitempty
  • 若字段没写 json 标签,Get("json") 返回空字符串,但字段名仍可能被默认使用(取决于 encoder 实现)

json.Marshal 底层是否真用反射遍历字段

是,但不是每次调用都现场反射。Go 的 json 包对每个 struct 类型做了一次反射缓存:首次序列化某类型时,marshal 函数调用 typeFields 构建字段索引表(含名称、偏移、标签解析结果),存进 structType 对应的 structCache 全局 map 中。

后续同类型的 Marshal 直接查缓存,跳过反射开销。这也是为什么首次调用稍慢,之后极快的原因。

立即学习go语言免费学习笔记(深入)”;

注意:json.Encoder.Encode 复用同一套缓存逻辑,不重复构建;但自定义 MarshalJSON 方法会绕过字段缓存,直接调用你实现的逻辑。

自定义 marshal 时如何复用原生 JSON 标签逻辑

如果你实现了 MarshalJSON(),又想保留字段级 json 标签行为(比如只过滤某些字段,其余照常),别自己再 parse tag——直接调用 json.Marshal 对子结构操作更安全。

func (u User) MarshalJSON() ([]byte, error) {     type Alias User // 防止无限递归     if u.SkipSensitive {         return json.Marshal(struct {             Name  string `json:"name"`             Email string `json:"email,omitempty"`         }{             Name:  u.Name,             Email: u.Email,         })     }     return json.Marshal(Alias(u)) }

这里用 type Alias User 断开递归,再嵌套匿名结构体控制输出字段,完全复用标准库的标签解析和编码逻辑。硬编码字段名比运行时反射取 tag 更可控,也避免误读 omitempty 等语义。

反射读 tag 后修改字段值会不会影响 JSON 输出

不会。反射读取 json 标签只是获取元信息,不影响序列化行为本身。真正决定输出内容的是 json.Marshal 内部的字段遍历逻辑 + 缓存结构 + 当前值。

但要注意:如果你用反射修改了 struct 字段值(比如把 int 改成 0),那 Marshal 输出自然会变——这是值变了,不是 tag 被动了。

容易混淆的点:有人试图用反射“动态覆盖”某个字段的 json 标签,比如改 StructField.Tag 字符串。这无效:reflect.StructField 是只读副本,改它不影响类型系统,也不触发缓存更新。

text=ZqhQzanResources