Go 語言 JSON 標(biāo)準(zhǔn)庫的使用
Go 語言中的 encoding/json 庫提供了復(fù)雜的將 Go 中各種類型與JSON格式之間轉(zhuǎn)換的功能, 我們主要使用以下幾個(gè)功能:
- 將一個(gè)切片、結(jié)構(gòu)體或字典序列化成 JSON 格式的字符串【字節(jié)流】。
- 將一個(gè) JSON 格式的字符串【字節(jié)流】反序列化成一個(gè)切片、結(jié)構(gòu)體或字典。
序列化
序列化使用 json 庫中的Marshal函數(shù):
func Marshal(v interface{}) ([]byte, error)
1. 結(jié)構(gòu)體序列化
比如使用以下的結(jié)構(gòu)體表示一部電影:
type Movie struct {
Title string
Year int `json:"released"`
Color bool `json:"color,omitempty"`
Actors []string
}
定義里類型后面跟的字符串 json:"released"和json:"color,omitempty,稱為 field tags,它告訴 json 庫在執(zhí)行序列化時(shí)的一些規(guī)則:
json:"released"使得在序列化后對(duì)應(yīng)的名字為"released",而不是"Year"。json:"color,omitempty"使得如果 Color 成員的值為false,那么就忽略它,不對(duì)它進(jìn)行序列化。
沒有 field tags 時(shí)進(jìn)行序列化的一些規(guī)則:
- 如果結(jié)構(gòu)體成員的名字不是以大寫字母開頭,則不對(duì)它進(jìn)行序列化。
- 如果結(jié)構(gòu)體成員的名字是以大寫字母開頭,則序列化后的名字就是成員名。
進(jìn)行序列化的代碼如下:
movie := Movie{
Title: "Casablanca",
Year: 1942,
Color: false,
Actors: []string{"Humphrey Bogart", "Ingrid Bergman"},
}
data, err := json.Marshal(movie)
if err != nil {
log.Fatalf("JSON marshaling failed: %s", err)
}
fmt.Printf("%s\n", data)
輸出:
{"Title":"Casablanca","released":1942,"Actors":["Humphrey Bogart","Ingrid Bergman"]}
2. 字典序列化
一個(gè)字典要想序列化成- JSON 格式,它的 key 必須是字符串。
以下是一個(gè)例子:
info := map[string]int{
"width": 1280,
"height": 720,
}
data, err := json.MarshalIndent(info, "", " ")
if err != nil {
log.Fatalf("JSON marshaling failed: %s", err)
}
fmt.Printf("%s\n", data)
輸出:
{
"height": 720,
"width": 1280
}
這里我們使用MarshalIndent函數(shù),使得輸出后的 JSON 格式更易閱讀。序列化后的名字就是字典中 key 的名稱。
3. 切片序列化
直接看一個(gè)例子:
type Movie struct {
Title string
Year int `json:"released"`
Color bool `json:"color,omitempty"`
Actors []string
}
var movies = []Movie{
{
Title: "Casablanca",
Year: 1942,
Color: false,
Actors: []string{"Humphrey Bogart", "Ingrid Bergman"},
},
{
Title: "Cool Hand Luke",
Year: 1967,
Color: true,
Actors: []string{"Paul Newman"},
},
{
Title: "Bullitt",
Year: 1968,
Color: true,
Actors: []string{"Steve McQueen", "Jacqueline Bisset"},
},
}
data, err := json.MarshalIndent(movies, "", " ")
if err != nil {
log.Fatalf("JSON marshaling failed: %s", err)
}
fmt.Printf("%s\n", data)
輸出:
[
{
"Title": "Casablanca",
"released": 1942,
"Actors": [
"Humphrey Bogart",
"Ingrid Bergman"
]
},
{
"Title": "Cool Hand Luke",
"released": 1967,
"color": true,
"Actors": [
"Paul Newman"
]
},
{
"Title": "Bullitt",
"released": 1968,
"color": true,
"Actors": [
"Steve McQueen",
"Jacqueline Bisset"
]
}
]
反序列化
反序列化使用Unmarshal函數(shù):
func Unmarshal(data []byte, v interface{}) error
1. 明確知道 JSON 格式
我們要先將 JSON 格式表示為一個(gè)確定的類型。
1.如下的 JSON 格式:
{
"name": "Awesome 4K",
"resolutions": [
{
"width": 1280,
"height": 720
},
{
"width": 1920,
"height": 1080
},
{
"width": 3840,
"height": 2160
}
]
}
我們可以用如下結(jié)構(gòu)體來表示它:
struct {
Name string
Resolutions []struct {
Width int
Height int
}
}
2.如下的 JSON 格式:
{
"height": 720,
"width": 1280
}
也可以使用map[string]int,也就是字典來表示。
3.如下的 JSON 格式:
[
{
"width": 1280,
"height": 720
},
{
"width": 1920,
"height": 1080
},
{
"width": 3840,
"height": 2160
}
]
使用切片[]map[string]int來表示。
不管怎樣,一個(gè)確定的JSON格式總是可以使用切片、結(jié)構(gòu)體或字典來表示。
之后就可以執(zhí)行反序列化了:
var jsonBlob = []byte(`
[
{
"width": 1280,
"height": 720
},
{
"width": 1920,
"height": 1080,
},
{
"width": 3840,
"height": 2160
}
]
`)
di := []map[string]int{}
err = json.Unmarshal(jsonBlob, &di)
if err != nil {
fmt.Println("error:", err)
}
fmt.Printf("%+v\n", di)
輸出
[map[height:720 width:1280] map[height:1080 width:1920] map[height:2160 width:3840]]
2. 無法確定 JSON 格式
無法確定的格式可以直接使用interface{}類型來表示。這個(gè)時(shí)候,如果是JSON對(duì)象,則反序列化時(shí) json 庫會(huì)使用map[string]interface{}類型來表示它。如果是JSON數(shù)組,則會(huì)使用[]interface{}類型來表示它。具體interface{}對(duì)應(yīng)的具體類型,就需要使用類型斷言了。
具體看一個(gè)示例:
package main
import (
"encoding/json"
"fmt"
)
func main() {
var jsonBlob = []byte(`
{
"name": "Awesome 4K",
"price": 1999.9,
"resolutions": [
{
"width": 1280,
"height": 720
},
{
"width": 1920,
"height": 1080
},
{
"width": 3840,
"height": 2160
}
]
}
`)
var d interface{}
err := json.Unmarshal(jsonBlob, &d)
if err != nil {
fmt.Println("error:", err)
}
fmt.Println(d)
m := d.(map[string]interface{})
for k, v := range m {
switch vv := v.(type) {
case string:
fmt.Println(k, "is string", vv)
case float64:
fmt.Println(k, "is float64", vv)
case []interface{}:
fmt.Println(k, "is an array:")
for i, u := range vv {
fmt.Println(i, u)
}
default:
fmt.Println(k, "is of a type I don't know how to handle")
}
}
}
輸出:
map[name:Awesome 4K price:1999.9 resolutions:[map[height:720 width:1280] map[height:1080 width:1920] map[height:2160 width:3840]]]
resolutions is an array:
0 map[height:720 width:1280]
1 map[height:1080 width:1920]
2 map[height:2160 width:3840]
name is string Awesome 4K
price is float64 1999.9
到此這篇關(guān)于Go 語言 JSON 標(biāo)準(zhǔn)庫的使用的文章就介紹到這了,更多相關(guān)Go 語言 JSON 標(biāo)準(zhǔn)庫內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
golang架構(gòu)設(shè)計(jì)開閉原則手寫實(shí)現(xiàn)
這篇文章主要為大家介紹了golang架構(gòu)設(shè)計(jì)開閉原則手寫實(shí)例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
Go語言映射內(nèi)部實(shí)現(xiàn)及基礎(chǔ)功能實(shí)戰(zhàn)
這篇文章主要為大家介紹了Go語言映射的內(nèi)部實(shí)現(xiàn)和基礎(chǔ)功能實(shí)戰(zhàn),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪<BR>2022-03-03
Go語言學(xué)習(xí)之new函數(shù)的用法詳解
這篇文章主要為大家詳細(xì)介紹了Go語言中new()函數(shù)的相關(guān)知識(shí)以及具體用法,文中的示例代碼講解詳細(xì),具有一定的學(xué)習(xí)價(jià)值,感興趣的小伙伴可以了解一下2023-05-05
在Linux系統(tǒng)中安裝Go語言的詳細(xì)教程
這篇文章主要介紹了在Linux系統(tǒng)中安裝Go語言的詳細(xì)教程,由于國(guó)內(nèi)很多人對(duì)谷歌的盲目追捧,導(dǎo)致Go語言在國(guó)內(nèi)的人氣遠(yuǎn)超國(guó)外...需要的朋友可以參考下2015-06-06
Go語言中Struct與繼承與匿名字段和內(nèi)嵌結(jié)構(gòu)體全面詳解
這篇文章主要介紹了Go語言中Struct與繼承與匿名字段和內(nèi)嵌結(jié)構(gòu)體,Go語言中通過結(jié)構(gòu)體的內(nèi)嵌再配合接口比面向?qū)ο缶哂懈叩臄U(kuò)展性和靈活性,感興趣的可以了解一下2023-04-04
Go語言標(biāo)準(zhǔn)庫sync.Once使用場(chǎng)景及性能優(yōu)化詳解
這篇文章主要為大家介紹了Go語言標(biāo)準(zhǔn)庫sync.Once使用場(chǎng)景及性能優(yōu)化詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-12-12
Go?通過?Map/Filter/ForEach?等流式?API?高效處理數(shù)據(jù)的思路詳解
Stream?的實(shí)現(xiàn)思想就是將數(shù)據(jù)處理流程抽象成了一個(gè)數(shù)據(jù)流,每次加工后返回一個(gè)新的流供使用。這篇文章主要介紹了Go?通過?Map/Filter/ForEach?等流式?API?高效處理數(shù)據(jù),需要的朋友可以參考下2022-01-01

