我們?cè)趯?duì)外提供API接口,返回響應(yīng)的時(shí)候,很多時(shí)候需要使用如下的數(shù)據(jù)結(jié)構(gòu)
type Response struct {
Code int `json:"code"`
Msg string `json:"msg"`
Data interface{} `json:"data"`
}
該API接口返回一個(gè)狀體碼,狀態(tài)信息,以及具體的值。但是具體的值可能根據(jù)各個(gè)接口的不同而不同。
在實(shí)際的開發(fā)過程中我們可能會(huì)得到一個(gè)實(shí)際的數(shù)據(jù)值,并將這個(gè)值賦值給data,然后json序列化返回給調(diào)用方。
這時(shí)如果你得到的data是一個(gè)經(jīng)過json序列化之后的字符串,類似于{"Name":"happy"},然后再將這個(gè)字符串賦值給data,此時(shí)將response序列化得到的string,如下
{“code”:1,”msg”:”success”,”data”:”{\”Name\”:\”Happy\”}”}
我們會(huì)發(fā)現(xiàn)之前已經(jīng)序列化好的字符串,每一個(gè)都多了\,這是因?yàn)檗D(zhuǎn)義引起的問題。
解決方法
直接將未序列化的data賦值給data
package main
import (
"encoding/json"
"fmt"
)
type Response struct {
Code int `json:"code"`
Msg string `json:"msg"`
Data interface{} `json:"data"`
}
type People struct {
Name string
}
func main() {
data := People{Name: "happy"}
response := Response{
Code: 1,
Msg: "success",
Data: data,
}
b, err := json.Marshal(response)
if err != nil {
fmt.Println("err", err)
}
fmt.Println(string(b))
}
使用json 的RawMessage 將轉(zhuǎn)義后的string,賦值給data
package main
import (
"encoding/json"
"fmt"
)
type Response struct {
Code int `json:"code"`
Msg string `json:"msg"`
Data interface{} `json:"data"`
}
type People struct {
Name string
}
func main() {
data := `{"Name":"Happy"}`
response := Response{
Code: 1,
Msg: "success",
Data: json.RawMessage(data),
}
b, err := json.Marshal(response)
if err != nil {
fmt.Println("err", err)
}
fmt.Println(string(b))
}
通過使用json的json.RawMessage(data)函數(shù)將其轉(zhuǎn)換一下,這樣也能保證不存在轉(zhuǎn)義符。
補(bǔ)充:golang string轉(zhuǎn)json的一些坑
先帶來點(diǎn)冷知識(shí),不知道大家知不知道,反正我剛知道...
大佬們都知道怎么在string中給string類型賦值帶雙引號(hào)的字符串,沒錯(cuò)就是用反斜杠,如下:
msg := "{\"name\":\"zhangsan\", \"age\":18, \"id\":122463, \"sid\":122464}"
但是golang還支持另外一個(gè)符號(hào),我初學(xué)時(shí)候以為是單引號(hào),但其實(shí)不是,是esc鍵下邊那個(gè),那么賦值帶雙引號(hào)的字符串就如下就行了:
ret := `{"access_token":"uAUS6o5g-9rFWjYt39LYa7TKqiMVsIfCGPEN4IZzdAk5-T-ryVhL7xb8kYciuU_m","expires_in":7200}`
進(jìn)入正題
先看一段代碼,起作用是把字符串轉(zhuǎn)換為結(jié)構(gòu)體對(duì)應(yīng)的json
type people struct {
name string `json:"name"`
age int `json:"age"`
id int `json:"id"`
}
type student struct {
people
id int `json:"sid"`
}
func main() {
msg := "{\"name\":\"zhangsan\", \"age\":18, \"id\":122463, \"sid\":122464}"
var someOne student
if err := json.Unmarshal([]byte(msg), someOne); err == nil {
fmt.Println(someOne)
fmt.Println(someOne.people)
} else {
fmt.Println(err)
}
}
仔細(xì)看看,有沒有錯(cuò)?我只能說,這樣是輸出不出來答案的,賦值錯(cuò)誤,看下面的運(yùn)行結(jié)果:

傷腦筋啊,我仔細(xì)看了半天,發(fā)現(xiàn)在定義的people和student兩個(gè)結(jié)構(gòu)體下邊有綠色的波浪線(我用的vscode),像下邊這樣:

鼠標(biāo)放上去顯示的是:

大家都知道,golang中變量聲明成大寫和小寫能引用的范圍是不一樣的,那我就想了,大小寫問題???一臉懵逼把變量名首字母改成了大寫,然后...就行了,代碼變成了下邊這樣:
type people struct {
Name string `json:"name"`
Age int `json:"age"`
ID int `json:"id"`
}
type student struct {
people
ID int `json:"sid"`
}
func main() {
msg := "{\"name\":\"zhangsan\", \"age\":18, \"id\":122463, \"sid\":122464}"
var someOne student
if err := json.Unmarshal([]byte(msg), someOne); err == nil {
fmt.Println(someOne)
fmt.Println(someOne.people)
} else {
fmt.Println(err)
}
}
輸出的結(jié)果這樣:

以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
您可能感興趣的文章:- golang json數(shù)組拼接的實(shí)例
- 在golang xorm中使用postgresql的json,array類型的操作
- golang中json小談之字符串轉(zhuǎn)浮點(diǎn)數(shù)的操作
- golang 實(shí)現(xiàn)json類型不確定時(shí)的轉(zhuǎn)換
- golang中json的omitempty使用操作
- golang:json 反序列化的[]和nil操作
- golang中json和struct的使用說明