Go語(yǔ)言內(nèi)建對(duì)JSON的支持。使用Go語(yǔ)言內(nèi)置的encoding/json標(biāo)準(zhǔn)庫(kù),開(kāi)發(fā)者可以輕松使用Go程序生成和解析JSON格式的數(shù)據(jù)。在Go語(yǔ)言實(shí)現(xiàn)JSON的編碼和解碼時(shí),遵循RFC4627協(xié)議標(biāo)準(zhǔn)。
1.編碼為JSON格式
使用json.Marshal()函數(shù)可以對(duì)一組數(shù)據(jù)進(jìn)行JSON格式的編碼。json.Marshal()函數(shù)的聲明如下:
假如有如下一個(gè)Book類(lèi)型的結(jié)構(gòu)體:
并且有如下一個(gè)Book類(lèi)型的實(shí)例對(duì)象:
然后,我們可以使用json.Marshal()函數(shù)將gobook實(shí)例生成一段JSON格式的文本:
如果編碼成功,err將賦于零值nil,變量b將會(huì)是一個(gè)進(jìn)行JSON格式化后的[]byte類(lèi)型:
當(dāng)我們調(diào)用json.Marshal(gobook)語(yǔ)句時(shí),會(huì)遞歸遍歷gobool對(duì)象,如果發(fā)現(xiàn)gobook這個(gè)數(shù)據(jù)結(jié)構(gòu)實(shí)現(xiàn)了json.Marshaler接口且包含有效的值,Marshal()就會(huì)調(diào)用其MarshalJSON()方法將該數(shù)據(jù)結(jié)構(gòu)生成JSON格式的文本。
Go語(yǔ)言的大多數(shù)數(shù)據(jù)類(lèi)型都可以轉(zhuǎn)化為有效的JSON文本,但channel、complex和函數(shù)這幾種類(lèi)型除外。
如果轉(zhuǎn)化前的數(shù)據(jù)結(jié)構(gòu)中出現(xiàn)指針,那么將會(huì)轉(zhuǎn)換指針?biāo)傅闹?,如果指針指向的是零值,那么nill將作為轉(zhuǎn)化后的結(jié)果輸出。
在Go中,JSON轉(zhuǎn)換前后的數(shù)據(jù)類(lèi)型映射如下:
·布爾值轉(zhuǎn)換為JSON后還是布爾類(lèi)型。
·浮點(diǎn)數(shù)和整數(shù)會(huì)被轉(zhuǎn)換為JSON里面的常規(guī)數(shù)字。
·字符串將以UTF-8編碼轉(zhuǎn)換輸出為Unicode字符集的字符串,特殊字符比如將會(huì)被轉(zhuǎn)換為\u003c。
·數(shù)組和切片會(huì)轉(zhuǎn)換為JSON里面的數(shù)組,但[]btye類(lèi)型的值將被轉(zhuǎn)換為Base64編碼后的字符串,slice類(lèi)型額零值會(huì)被轉(zhuǎn)換為null。
·結(jié)構(gòu)體會(huì)轉(zhuǎn)換為JSON對(duì)象,并且只有結(jié)構(gòu)體里面以大寫(xiě)字母開(kāi)頭的可被導(dǎo)出的字段才會(huì)被轉(zhuǎn)換輸出,而這些可導(dǎo)出的字段會(huì)作為JSON對(duì)象的字符串索引。
·轉(zhuǎn)換一個(gè)map類(lèi)型的數(shù)據(jù)結(jié)構(gòu)時(shí),該數(shù)據(jù)的類(lèi)型必須是map[string]T(T可以是encoding/json包支持的任意數(shù)據(jù)類(lèi)型)。
2.解碼JSON數(shù)據(jù)
可以使用json.Unmarshal()函數(shù)將JSON格式的文本解碼為Go里面預(yù)期的數(shù)據(jù)結(jié)構(gòu)。json.Unmarshal()函數(shù)的原型如下:
該函數(shù)的第一次參數(shù)是輸入,即JSON格式的文本(比特序列),第二個(gè)參數(shù)表示目標(biāo)輸出容器,用于存放解碼后的值。
要解碼一段JSON數(shù)據(jù),首先需要在Go中創(chuàng)建一個(gè)目標(biāo)類(lèi)型的實(shí)例對(duì)象,用于存放解碼后的值:
var book BOOK
然后調(diào)用json.Unmarshal()函數(shù),將[]byte類(lèi)型的JSON數(shù)據(jù)作為第一個(gè)參數(shù)傳入,將book實(shí)例變量的指針作為第二個(gè)參數(shù)傳入:
err := json.Unmarshal(b, book)
如果b是一個(gè)有效的JSON數(shù)據(jù)并能和book結(jié)構(gòu)對(duì)應(yīng)起來(lái),那么JSON解碼后的值將會(huì)一一存放到book結(jié)構(gòu)體中。解碼成功后的book數(shù)據(jù)如下:
json.Unmarshal()函數(shù)會(huì)根據(jù)一個(gè)約定的順序查找目標(biāo)結(jié)構(gòu)中的字段,如果找到一個(gè)即發(fā)生匹配。假設(shè)一個(gè)JSON對(duì)象有個(gè)名為"Foo"的索引,要將"Foo"所對(duì)應(yīng)的值填充到目標(biāo)結(jié)構(gòu)體的目標(biāo)字段上,json.Unmarshal()將會(huì)遵循如下順序進(jìn)行查找匹配:
(1)一個(gè)包含F(xiàn)oo標(biāo)簽的字段;
(2)一個(gè)名為Foo的字段;
(3)一個(gè)名為Foo或者除了首字母其他字母不區(qū)分大小的命名為Foo的字段。
這些字段在類(lèi)型聲明中必須都是以大寫(xiě)字母開(kāi)頭、可被導(dǎo)出的字段。
如果JSON中的字段在Go目標(biāo)類(lèi)型中不存在,json.Unmarshal()函數(shù)在解碼過(guò)程中會(huì)丟棄該字段。
3.解碼未知結(jié)構(gòu)的JSON數(shù)據(jù)
Go內(nèi)建了靈活的類(lèi)型系統(tǒng),向我們傳達(dá)了一個(gè)很有價(jià)值的信息:空接口是通用類(lèi)型。如果要解碼一段未知結(jié)構(gòu)的JSON,只需將這段JSON數(shù)據(jù)解碼輸出到一個(gè)空接口即可。在解碼JSON數(shù)據(jù)的過(guò)程中,JSON數(shù)據(jù)里面的元素類(lèi)型將做如下轉(zhuǎn)換:
·JSON中的布爾值將會(huì)轉(zhuǎn)換為Go中的bool類(lèi)型;
·數(shù)值會(huì)被轉(zhuǎn)換為Go中的float64類(lèi)型;
·字符串轉(zhuǎn)換后還是string類(lèi)型;
·JSON數(shù)組會(huì)轉(zhuǎn)換為[]interface{}類(lèi)型;
·JSON對(duì)象會(huì)轉(zhuǎn)換為map[string]interface{}類(lèi)型;
·null值會(huì)被轉(zhuǎn)換為nil.
在Go的標(biāo)準(zhǔn)庫(kù)encoding/json包中,允許使用map[string] interface{}和[]interface{}類(lèi)型的值來(lái)分別存放未知結(jié)構(gòu)的JSON對(duì)象或數(shù)組,實(shí)例代碼如下:
在上述代碼中,r被定義為一個(gè)空接口。json.Unmarshal()函數(shù)將一個(gè)JSON對(duì)象解碼到空接口r中,最終r將會(huì)是一個(gè)鍵值對(duì)的map[string] interface{}結(jié)構(gòu):
要訪問(wèn)解碼后的數(shù)據(jù)結(jié)構(gòu),需要先判斷目標(biāo)結(jié)構(gòu)是否為預(yù)期的數(shù)據(jù)類(lèi)型:
然后,我們可以通過(guò)for循環(huán)搭配range語(yǔ)句一一訪問(wèn)解碼后的目標(biāo)數(shù)據(jù):
4.JSON的流式讀寫(xiě)
Go內(nèi)建的encoding/json包還提供Decoder和Encoder兩個(gè)類(lèi)型,用于支持JSON數(shù)據(jù)的流式,并提供NewDecoder()和NewEncoder()兩個(gè)函數(shù)來(lái)便于具體實(shí)現(xiàn):
使用Decoder 和Encoder對(duì)數(shù)據(jù)流進(jìn)行處理可以應(yīng)用得更為廣泛些,比如讀寫(xiě) HTTP 連接、WebSocket或文件等, Go的標(biāo)準(zhǔn)庫(kù)net/rpc/jsonrpc就是一個(gè)應(yīng)用了Decoder和Encoder的實(shí)際例子。
實(shí)例代碼:把sql結(jié)果集以json格式輸出
func getJSON(sqlString string) (string, error) {
stmt, err := db.Prepare(sqlString)
if err != nil {
return nil, err
}
defer stmt.Close()
rows, err := stmt.Query()
if err != nil {
return nil, err
}
defer rows.Close()
columns, err := rows.Columns()
if err != nil {
return "", err
}
count := len(columns)
tableData := make([]map[string]interface{}, 0)
values := make([]interface{}, count)
valuePtrs := make([]interface{}, count)
for rows.Next() {
for i := 0; i count; i++ {
valuePtrs[i] = values[i]
}
rows.Scan(valuePtrs...)
entry := make(map[string]interface{})
for i, col := range columns {
var v interface{}
val := values[i]
b, ok := val.([]byte)
if ok {
v = string(b)
} else {
v = val
}
entry[col] = v
}
tableData = append(tableData, entry)
}
jsonData, err := json.Marshal(tableData)
if err != nil {
return "", err
}
fmt.Println(string(jsonData))
return string(jsonData), nil
}
以上就是本文的全部?jī)?nèi)容了,希望對(duì)大家學(xué)習(xí)golang有所幫助
您可能感興趣的文章:- 跨域解決之JSONP和CORS的詳細(xì)介紹
- JSON生成Form表單的方法示例
- $.ajax中contentType: “application/json” 的用法詳解
- 簡(jiǎn)單說(shuō)說(shuō)angular.json文件的使用
- JavaScript根據(jù)json生成html表格的示例代碼
- MongoDB使用mongoexport和mongoimport命令,批量導(dǎo)出和導(dǎo)入JSON數(shù)據(jù)到同一張表的實(shí)例
- nodejs讀取本地中文json文件出現(xiàn)亂碼解決方法
- JavaScript JSON使用原理及注意事項(xiàng)