Go中JSON解析不抛异常而返回error,须用if err != nil检查;可类型断言json.SyntaxError或json.UnmarshalTypeError定位问题;推荐预校验json.Valid()、用omitempty标签和json.RawMessage提升容错。
在 Go 中解析 JSON 时,encoding/json 包不会抛出传统意义上的“异常”,而是通过返回 error 值来表示失败。正确处理这些错误是保障程序健壮性的关键。
常见错误包括:结构体字段标签不匹配、JSON 数据格式非法(如缺少引号、逗号错误)、类型不兼容(如把字符串当数字解析)、嵌套结构缺失或类型错位等。这些都会导致 json.Unmarshal 返回非 nil 的 error。
不要忽略 err 参数——这是最常见也最危险的疏忽。
json.Unmarshal(data, &v); if err != nil { ... }(err 未声明或作用域错误)if err := json.Unmarshal(data, &v); err != nil { log.Printf("JSON 解析失败: %v", err) }
Go 标准库的 json 错误没有导出具体类型,但你可以借助错误文本或封装辅助函数增强可读性:
立即学习“go语言免费学习笔记(深入)”;
strings.Contains(err.Error(), "invalid character") → 通常是非法 JSON 字符(如中文逗号、BOM 头、控制字符)strings.Contains(err.Error(), "cannot unmarshal") → 类型不匹配(如期望 float64 却给了 string)json.SyntaxError 或 json.UnmarshalTypeError 类型断言获取结构化信息例如:
if syntaxErr, ok := err.(*json.SyntaxError); ok {
log.Printf("JSON 语法错误,位置 %d: %v", syntaxErr.Offset, syntaxErr.Error())
}
if typeErr, ok := err.(*json.UnmarshalTypeError); ok {
log.Printf("类型错误:期望 %s,但得到 %s(字段 %s)",
typeErr.Type, typeErr.Value, typeErr.Field)
}
对不可控输入(如 API 请求),可组合多种策略降低崩溃风险:
json.Valid() 快速判断是否为合法 JSON 字节流(不解析,仅语法检查)omitempty 标签,允许缺失字段UnmarshalJSON 方法中实
现自定义反序列化)json.RawMessage 延迟解析,避免早期失败不复杂但容易忽略。核心就一条:把 error 当第一等公民对待,而不是补丁式地事后处理。