登录
首页 >  Golang >  Go教程

Golang JSON反序列化教程详解

时间:2026-05-22 14:11:24 433浏览 收藏

本文深入剖析了Golang中JSON反序列化(json.Unmarshal)的四大核心陷阱:必须传结构体指针否则解析结果全为零值;结构体字段需显式添加json tag才能正确映射大小写不一致或下划线命名的JSON键,标准库不支持自动驼峰转换;类型不匹配(如null赋值给非指针string、数字进string等)会直接报错,忽略错误检查将导致静默失败;以及性能隐患——重复解析同一段数据应改用json.RawMessage或中间map延迟解析。这些看似细微的细节,恰恰是生产环境中JSON解析崩溃、数据丢失却难以定位的根源。

Golang json.Unmarshal怎么用_Golang JSON反序列化教程【秒懂】

json.Unmarshal 传入的必须是指针

json.Unmarshal 一个普通变量(比如 var u User),它不会报错,但解析结果永远是零值。因为 Go 是值传递,函数内部只能修改副本。

正确做法永远是传地址:&u。哪怕结构体字段全是导出的、类型完全匹配,不传指针就白解析。

  • 错误写法:json.Unmarshal(data, u) → 字段全为零值
  • 正确写法:json.Unmarshal(data, &u)
  • 如果解析到 map 或 slice,也要传指针:json.Unmarshal(data, &m)json.Unmarshal(data, &s)

struct 字段没加 json tag 就无法反序列化

Go 的 json 包默认只处理导出字段(首字母大写),且要求字段名与 JSON key 完全一致(大小写敏感)。实际 JSON 常用小写下划线命名(如 user_name),不加 tag 就对不上。

别指望“自动转驼峰”——标准库不干这事,第三方库才可能支持。

  • JSON 中是 "user_id",结构体字段得写 UserID int `json:"user_id"`
  • 字段想忽略就用 json:"-";想允许空字符串或 null 赋值给 string,用 json:",omitempty"(注意:它只影响序列化,不影响反序列化)
  • 嵌套结构体字段也要逐层加 tag,漏一层就解析失败

反序列化时类型不匹配导致静默失败或 panic

json.Unmarshal 对类型不匹配的容忍度很低:数字进 string、bool 进 int、null 进非指针字段,都会直接返回 json.UnmarshalTypeError 错误。但很多人忽略 err 检查,结果变量还是零值,还以为数据丢了。

  • 常见错误现象:json: cannot unmarshal number into Go struct field X of type string
  • string 类型字段收到 null?必须改成 *string 才能接住(否则报错)
  • 整数字段收到浮点数(如 123.0)?Go 默认允许,但精度可能丢失;想严格校验就得先用 json.RawMessage 手动解析
  • 切片字段收到非数组值(比如 null 或对象)?会报错,不是变成空切片

性能和兼容性:不要反复 Unmarshal 同一段 data

每次调用 json.Unmarshal 都要重新解析整个字节流,开销不小。如果同一段 []byte 要解析成多个结构体(比如先看 type 字段再分发),别重复调用。

更稳妥的做法是先解析成 map[string]interface{}json.RawMessage,再按需转换。虽然内存多占一点,但避免了重复词法分析。

  • 高频场景(如 API 网关)建议用 json.RawMessage 延迟解析关键子字段
  • 注意:json.RawMessage 保存的是原始字节引用,如果原始 []byte 被复用或释放,它会失效
  • 标准库不支持流式解析(不像 Python 的 ijson),大 JSON 必须全加载进内存

最常被跳过的其实是错误检查和字段 tag 的一致性——写完 struct 不顺手补上 tag,或者把 err 写成 _,后面排查时就得一行行打日志。JSON 反序列化看着简单,崩起来悄无声息。

以上就是本文的全部内容了,是否有顺利帮助你解决问题?若是能给你带来学习上的帮助,请大家多多支持golang学习网!更多关于Golang的相关知识,也可关注golang学习网公众号。

资料下载
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>