在Go开发中,我们经常使用encoding/json
标准库进行JSON的编解码操作。但是,有时候我们可能会发现一个有趣的现象:同样是解析JSON字符到数字,有时会成功,而有时却会失败。这让我们很困惑,下面就来分析这个问题。
JSON和Go的数据类型映射
在开始之前,首先了解一下JSON和Go的数据类型是如何映射的。
- JSON的数值(无论整数还是浮点数)通常可以映射到Go的
int
、int64
、float64
等数值类型。 - JSON的字符串只能映射到Go的
string
类型。
简单来说,从类型的严格性来看,Go > JSON。
情况1:从JSON数值到Go数值
这种情况下,解析基本上是无问题的。因为JSON的数值类型可以适配Go的各种数值类型。
jsonStr := `{"number": 123}`
var obj struct { Number int `json:"number"` }
err := json.Unmarshal([]byte(jsonStr), &obj)
情况2:从JSON数值到Go字符串
当JSON字段是数值,而Go结构体中该字段被定义为字符串时,解析依然可以成功。
jsonStr := `{"number": 123}`
var obj struct { Number string `json:"number"` }
err := json.Unmarshal([]byte(jsonStr), &obj)
情况3:从JSON字符串到Go数值
这种情况下,解析通常会失败,并且返回类似json: cannot unmarshal string into Go struct field .number of type int
的错误。
jsonStr := `{"number": "123"}`
var obj struct { Number int `json:"number"` }
err := json.Unmarshal([]byte(jsonStr), &obj)
这就是错误json: cannot unmarshal string into Go struct field .number of type int
的典型场景。
为什么有时会成功?
我们可能偶尔发现,当我们尝试从JSON字符串解析到Go数值时,居然成功了。这很可能是因为在解析过程中进行了某种隐式转换或类型断言。
例如,如果我们用interface{}
来接收JSON字段,并且后续进行了手动的类型转换,那么解析可能会成功。
jsonStr := `{"number": "123"}`
var obj map[string]interface{}
err := json.Unmarshal([]byte(jsonStr), &obj)
number, ok := obj["number"].(string)
if ok {
intValue, _ := strconv.Atoi(number)
}
总结
在Go的JSON解析中,类型是非常严格的。正是因为这种严格性,我们才能写出类型安全的代码。但同时,这也意味着当JSON数据与Go结构体类型不匹配时,我们需要付出额外的努力进行处理。
如果你对更多的Go开发细节或相关话题感兴趣,请关注我们的博客,我们还有大量与企业管理、团队管理、项目管理和个人成长相关的精彩文章。
祝您编程愉快!