01
介绍
在使用 Go 语言调用三方RESTful
接口时,因为无法直接操作 json 字符串,所以我们需要先将 json 字符串转换为 map 或 struct。
本文我们介绍一下怎么处理三方接口返回数据。
普通 json
我们先看一下三方接口返回的普通 json。
func main() { // 三方返回普通 json 字符串 jsonRes := `{ "Id": 1001, "Name": "frank" }` data := new(User) err := json.Unmarshal([]byte(jsonRes), &data) if err != nil { log.Printf("json Unmarshal err:%v\n", err) return } fmt.Printf("data=%+v", data) } type User struct { Id int Name string }
输出结果:
data=&{Id:1001 Name:frank}
阅读上面这段代码,我们构造一个简单的 json 字符串,模拟三方接口返回数据。
我们使用 Go 标准库 encoding/json
的 Unmarshal
函数,可以很容易将 json 数据解码到 struct,从而方便我们读取返回数据。
但是,需要读者朋友们注意的是,假如三方接口返回数据的字段类型随机变化(比如示例中的 Id
字段,可能是整型或字符串随机返回),我们使用 Unmarshal
函数解码时,就有可能会返回错误,如下所示:
2022/08/15 14:07:41 json Unmarshal err:json: cannot unmarshal string into Go struct field User.Id of type int
阅读到这里,我相信已经有读者朋友们想到,可以把返回数据解码到 map[string]interface{}
类型的变量中。
示例代码:
func main() { // 三方返回普通 json 字符串 jsonRes := `{ "Id": 1001, "Name": "frank" }` data2 := make(map[string]interface{}) err := json.Unmarshal([]byte(jsonRes), &data2) if err != nil { log.Printf("json Unmarshal err:%v\n", err) return } fmt.Printf("data2=%+v", data2) }
输出结果:
data2=map[Id:1001 Name:frank]
阅读上面示例代码,我们可以通过将返回数据解码到 map[string]interface{}
类型的变量中,从而解决三方接口返回数据的字段类型不固定的问题。
普通 json 使用该方式处理确实可行,但是如果嵌套 json,也可以这么处理,但是读取嵌套 json 的子字段就不那么方便了。
03
嵌套 json
我们再构造一个三方接口返回数据是嵌套 json 的变量。
func main() { // 三方返回嵌套 json 字符串 jsonRes := `{ "Id": 1001, "Name": "frank", "Details": { "Gender": "man", "Age": 18, "Phone": "13800138000", "address": "Beijing" } }` data := new(User) err := json.Unmarshal([]byte(jsonRes), &data) if err != nil { log.Printf("json Unmarshal err:%v\n", err) return } fmt.Printf("data=%+v", data) } type User struct { Id int Name string Details Details } type Details struct { Gender string Age int Phone string Address string }
输出结果:
data=&{Id:1001 Name:frank Details:{Gender:man Age:18 Phone:13800138000 Address:Beijing}}
阅读上面这段代码,我们构造一个嵌套 json,使用 Unmarshal
函数解码到 struct 中。
但是,如果返回数据中的 Age
字段是字符串类型,我们使用 Unmarshal
函数解码时,就会返回以下错误:
2022/08/15 17:33:08 json Unmarshal err:json: cannot unmarshal string into Go struct field Details.Details.Age of type int
虽然,我们可以使用普通 json 中的处理方式,将返回数据解码到 map[string]interface{}
类型的变量中。但是,如果我们想要读取内嵌 json 中的子字段,就读取不到了。
怎么解决这个问题呢?我们可以借助三方库 mapstructure
,使用该三方库的 Decode
函数替代 Go 标准库 encoding/json
的 Unmarshal
函数。
示例代码:
func main() { // 三方返回嵌套 json 字符串 jsonRes := `{ "Id": 1001, "Name": "frank", "Details": { "Gender": "man", "Age": "18", "Phone": "13800138000", "address": "Beijing" } }` tmpData := make(map[string]interface{}) err := json.Unmarshal([]byte(jsonRes), &tmpData) if err != nil { log.Printf("json Unmarshal err:%v\n", err) return } data2 := new(User) err = mapstructure.Decode(tmpData, data2) if err != nil { log.Printf("decode err:%v\n", err) return } fmt.Printf("data2=%+v\n", data2) fmt.Printf("age=%v\n", data2.Details.Age) } type User struct { Id int Name string Details Details } type Details struct { Gender string Age interface{} Phone string Address string }
输出结果:
data2=&{Id:1001 Name:frank Details:{Gender:man Age:18 Phone:13800138000 Address:Beijing}} age=18
阅读上面这段代码,我们将嵌套 struct 的 Age
字段定义为 interface{}
类型,首先,我们使用 Go 标准库的 Unmarshal
函数将返回数据解码到 map[string]interface{}
类型的变量中。
然后使用三方库 mapstructure
的 Decode
函数,将 map[string]interface{}
类型的变量中的数据解码到 struct 中,从而实现我们可以读取内嵌 json 中的子字段。
三方库 mapstructure
还有很多其他好用的功能,感兴趣的读者朋友们可以阅读官方文档了解更多内容。
04
总结
本文我们主要介绍怎么处理三方接口返回数据,其中包含普通 json 和嵌套 json,在处理嵌套 json 的内容中,我们介绍了三方库 mapstructure
的简单使用方式。
建议读者朋友们阅读三方库 mapstructure
的官方文档,了解它提供的更多函数和方法。
推荐阅读:
参考资料: