您好,登錄后才能下訂單哦!
這篇文章主要介紹“Go中JSON處理方法是什么”,在日常操作中,相信很多人在Go中JSON處理方法是什么問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”Go中JSON處理方法是什么”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!
Go語言內(nèi)建對JSON的支持。使用Go語言內(nèi)置的encoding/json標(biāo)準(zhǔn)庫,開發(fā)者可以輕松使用Go程序生成和解析JSON格式的數(shù)據(jù)。在Go語言實現(xiàn)JSON的編碼和解碼時,遵循RFC4627協(xié)議標(biāo)準(zhǔn)。
首先我們來看一下Go語言中json.Marshal()(系列化)與json.Unmarshal(反序列化)的基本用法。
type Person struct { Name string Age int64 Weight float64 } func main() { p1 := Person{ Name: "小明", Age: 18, Weight: 71.5, } // struct -> json string b, err := json.Marshal(p1)if err != nil { fmt.Printf("json.Marshal failed, err:%v\n", err)return} fmt.Printf("str:%s\n", b) // json string -> struct var p2 Person err = json.Unmarshal(b, &p2)if err != nil { fmt.Printf("json.Unmarshal failed, err:%v\n", err)return} fmt.Printf("p2:%#v\n", p2) }
輸出: str:{“Name”:”小明”,”Age”:18,”Weight”:71.5} p2:main.Person{Name:”小明”, Age:18, Weight:71.5}
Tag是結(jié)構(gòu)體的元信息,可以在運行的時候通過反射的機(jī)制讀取出來。 Tag在結(jié)構(gòu)體字段的后方定義,由一對反引號包裹起來,具體的格式如下:
`key1:"value1" key2:"value2"`
結(jié)構(gòu)體tag由一個或多個鍵值對組成。鍵與值使用冒號分隔,值用雙引號括起來。 同一個結(jié)構(gòu)體字段可以設(shè)置多個鍵值對tag,不同的鍵值對之間使用空格分隔。
序列化與反序列化默認(rèn)情況下使用結(jié)構(gòu)體的字段名,我們可以通過給結(jié)構(gòu)體字段添加tag來指定json序列化生成的字段名。
// 使用json tag指定序列化與反序列化時的行為type Person struct { Name string `json:"name"` // 指定json序列化/反序列化時使用小寫name Age int64 Weight float64 }
如果你想在json序列化/反序列化的時候忽略掉結(jié)構(gòu)體中的某個字段,可以按如下方式在tag中添加-。
// 使用json tag指定json序列化與反序列化時的行為type Person struct { Name string `json:"name"` // 指定json序列化/反序列化時使用小寫name Age int64 Weight float64 `json:"-"` // 指定json序列化/反序列化時忽略此字段 }
當(dāng) struct 中的字段沒有值時, json.Marshal() 序列化的時候不會忽略這些字段,而是默認(rèn)輸出字段的類型零值(例如int和float類型零值是 0,string類型零值是””,對象類型零值是 nil)。如果想要在序列序列化時忽略這些沒有值的字段時,可以在對應(yīng)字段添加omitempty tag。 舉個例子:
type User struct { Name string `json:"name"` Email string `json:"email"` Hobby []string `json:"hobby"` } func omitemptyDemo() { u1 := User{ Name: "小明", } // struct -> json string b, err := json.Marshal(u1)if err != nil { fmt.Printf("json.Marshal failed, err:%v\n", err)return} fmt.Printf("str:%s\n", b) }
輸出結(jié)果: str:{“name”:”小明”,”email”:””,”hobby”:null}
如果想要在最終的序列化結(jié)果中去掉空值字段,可以像下面這樣定義結(jié)構(gòu)體:
// 在tag中添加omitempty忽略空值 // 注意這里 hobby,omitempty 合起來是json tag值,中間用英文逗號分隔type User struct { Name string `json:"name"` Email string `json:"email,omitempty"` Hobby []string `json:"hobby,omitempty"` }
此時,再執(zhí)行上述的omitemptyDemo,輸出結(jié)果如下: str:{“name”:”小明”} // 序列化結(jié)果中沒有email和hobby字段
說句題外話,我們使用gorm操作數(shù)據(jù)庫的話,經(jīng)常會遇到想忽略指定字段修改的問題,比如結(jié)構(gòu)體中的關(guān)聯(lián)實體,只想json展示,form提交時忽略實體,這種問題我會單獨整理一篇出來。
首先來看幾種結(jié)構(gòu)體嵌套的示例:
type User struct { Name string `json:"name"` Email string `json:"email,omitempty"` Hobby []string `json:"hobby,omitempty"` Profile }type Profile struct { Website string `json:"site"` Slogan string `json:"slogan"` } func nestedStructDemo() { u1 := User{ Name: "小明", Hobby: []string{"足球", "籃球"}, } b, err := json.Marshal(u1)if err != nil { fmt.Printf("json.Marshal failed, err:%v\n", err)return} fmt.Printf("str:%s\n", b) }
匿名嵌套Profile時序列化后的json串為單層的: str:{“name”:”小明”,”hobby”:[“足球”,”藍(lán)球”],”site”:””,”slogan”:””}
想要變成嵌套的json串,需要改為具名嵌套或定義字段tag:
type User struct { Name string `json:"name"` Email string `json:"email,omitempty"` Hobby []string `json:"hobby,omitempty"` Profile `json:"profile"` } // str:{"name":"小明","hobby":["足球","籃球"],"profile":{"site":"","slogan":""}}
想要在嵌套的結(jié)構(gòu)體為空值時,忽略該字段,僅添加omitempty是不夠的:
type User struct { Name string `json:"name"` Email string `json:"email,omitempty"` Hobby []string `json:"hobby,omitempty"` Profile `json:"profile,omitempty"` } // str:{"name":"小明","hobby":["足球","籃球"],"profile":{"site":"","slogan":""}}
還需要使用嵌套的結(jié)構(gòu)體指針:
type User struct { Name string `json:"name"` Email string `json:"email,omitempty"` Hobby []string `json:"hobby,omitempty"` *Profile `json:"profile,omitempty"` //這里是重點 } // str:{"name":"小明","hobby":["足球","籃球"]}
我們需要json序列化User,但是不想把密碼也序列化,又不想修改User結(jié)構(gòu)體,這個時候我們就可以使用創(chuàng)建另外一個結(jié)構(gòu)體PublicUser匿名嵌套原User,同時指定Password字段為匿名結(jié)構(gòu)體指針類型,并添加omitemptytag,示例代碼如下:
type User struct { Name string `json:"name"` Password string `json:"password"` }type PublicUser struct { *User // 匿名嵌套 Password *struct{} `json:"password,omitempty"` } func omitPasswordDemo() { u1 := User{ Name: "小明", Password: "123456", } b, err := json.Marshal(PublicUser{User: &u1})if err != nil { fmt.Printf("json.Marshal u1 failed, err:%v\n", err)return} fmt.Printf("str:%s\n", b) // str:{"name":"小明"} }
有時候,前端在傳遞來的json數(shù)據(jù)中可能會使用字符串類型的數(shù)字,這個時候可以在結(jié)構(gòu)體tag中添加string來告訴json包從字符串中解析相應(yīng)字段的數(shù)據(jù):
type Card struct { ID int64 `json:"id,string"` // 添加string tag Score float64 `json:"score,string"` // 添加string tag } func intAndStringDemo() { jsonStr1 := `{"id": "1234567","score": "88.50"}` var c1 Cardif err := json.Unmarshal([]byte(jsonStr1), &c1); err != nil { fmt.Printf("json.Unmarsha jsonStr1 failed, err:%v\n", err)return} fmt.Printf("c1:%#v\n", c1) // c1:main.Card{ID:1234567, Score:88.5} }
到此,關(guān)于“Go中JSON處理方法是什么”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注億速云網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。