我們在對外提供API接口,返回響應(yīng)的時候,很多時候需要使用如下的數(shù)據(jù)結(jié)構(gòu)
type Response struct {
Code int `json:"code"`
Msg string `json:"msg"`
Data interface{} `json:"data"`
}
該API接口返回一個狀體碼,狀態(tài)信息,以及具體的值。但是具體的值可能根據(jù)各個接口的不同而不同。
在實際的開發(fā)過程中我們可能會得到一個實際的數(shù)據(jù)值,并將這個值賦值給data,然后json序列化返回給調(diào)用方。
這時如果你得到的data是一個經(jīng)過json序列化之后的字符串,類似于{"Name":"happy"},然后再將這個字符串賦值給data,此時將response序列化得到的string,如下
{“code”:1,”msg”:”success”,”data”:”{\”Name\”:\”Happy\”}”}
我們會發(fā)現(xiàn)之前已經(jīng)序列化好的字符串,每一個都多了\,這是因為轉(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)義符。
補充:golang string轉(zhuǎn)json的一些坑
先帶來點冷知識,不知道大家知不知道,反正我剛知道...
大佬們都知道怎么在string中給string類型賦值帶雙引號的字符串,沒錯就是用反斜杠,如下:
msg := "{\"name\":\"zhangsan\", \"age\":18, \"id\":122463, \"sid\":122464}"
但是golang還支持另外一個符號,我初學(xué)時候以為是單引號,但其實不是,是esc鍵下邊那個,那么賦值帶雙引號的字符串就如下就行了:
ret := `{"access_token":"uAUS6o5g-9rFWjYt39LYa7TKqiMVsIfCGPEN4IZzdAk5-T-ryVhL7xb8kYciuU_m","expires_in":7200}`
進入正題
先看一段代碼,起作用是把字符串轉(zhuǎn)換為結(jié)構(gòu)體對應(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)
}
}
仔細看看,有沒有錯?我只能說,這樣是輸出不出來答案的,賦值錯誤,看下面的運行結(jié)果:
![](/d/20211017/c5156d4bbd7288142f04d5bb7b26065b.gif)
傷腦筋啊,我仔細看了半天,發(fā)現(xiàn)在定義的people和student兩個結(jié)構(gòu)體下邊有綠色的波浪線(我用的vscode),像下邊這樣:
![](/d/20211017/6f2e8d72d37d4d171b5bc7359765109d.gif)
鼠標(biāo)放上去顯示的是:
![](/d/20211017/b699fc90c410e5e6de5659c182faa1ce.gif)
大家都知道,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é)果這樣:
![](/d/20211017/e8027249dc3bb08bc4f01d122949801b.gif)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。
您可能感興趣的文章:- golang json數(shù)組拼接的實例
- 在golang xorm中使用postgresql的json,array類型的操作
- golang中json小談之字符串轉(zhuǎn)浮點數(shù)的操作
- golang 實現(xiàn)json類型不確定時的轉(zhuǎn)換
- golang中json的omitempty使用操作
- golang:json 反序列化的[]和nil操作
- golang中json和struct的使用說明