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

反射怎么拿到 struct 字段的 json 标签值
Go 的 reflect.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 是只读副本,改它不影响类型系统,也不触发缓存更新。










