溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務(wù)條款》

如何在Go語言中處理JSON

發(fā)布時間:2022-04-15 14:02:03 來源:億速云 閱讀:552 作者:iii 欄目:編程語言

這篇文章主要介紹“如何在Go語言中處理JSON”,在日常操作中,相信很多人在如何在Go語言中處理JSON問題上存在疑惑,小編查閱了各式資料,整理出簡單好用的操作方法,希望對大家解答”如何在Go語言中處理JSON”的疑惑有所幫助!接下來,請跟著小編一起來學(xué)習(xí)吧!

1.編碼為JSON格式

   使用json.Marshal()函數(shù)可以對一組數(shù)據(jù)進行JSON格式的編碼。json.Marshal()函數(shù)的聲明如下:

如何在Go語言中處理JSON

   假如有如下一個Book類型的結(jié)構(gòu)體:

如何在Go語言中處理JSON

   并且有如下一個Book類型的實例對象:

如何在Go語言中處理JSON

    然后,我們可以使用json.Marshal()函數(shù)將gobook實例生成一段JSON格式的文本:

如何在Go語言中處理JSON

      如果編碼成功,err將賦于零值nil,變量b將會是一個進行JSON格式化后的[]byte類型:

如何在Go語言中處理JSON

    當(dāng)我們調(diào)用json.Marshal(gobook)語句時,會遞歸遍歷gobool對象,如果發(fā)現(xiàn)gobook這個數(shù)據(jù)結(jié)構(gòu)實現(xiàn)了json.Marshaler接口且包含有效的值,Marshal()就會調(diào)用其MarshalJSON()方法將該數(shù)據(jù)結(jié)構(gòu)生成JSON格式的文本。

    Go語言的大多數(shù)數(shù)據(jù)類型都可以轉(zhuǎn)化為有效的JSON文本,但channel、complex和函數(shù)這幾種類型除外。

    如果轉(zhuǎn)化前的數(shù)據(jù)結(jié)構(gòu)中出現(xiàn)指針,那么將會轉(zhuǎn)換指針?biāo)傅闹担绻羔樦赶虻氖橇阒担敲磏ill將作為轉(zhuǎn)化后的結(jié)果輸出。

    在Go中,JSON轉(zhuǎn)換前后的數(shù)據(jù)類型映射如下:

    ·布爾值轉(zhuǎn)換為JSON后還是布爾類型。

    ·浮點數(shù)和整數(shù)會被轉(zhuǎn)換為JSON里面的常規(guī)數(shù)字。

    ·字符串將以UTF-8編碼轉(zhuǎn)換輸出為Unicode字符集的字符串,特殊字符比如<將會被轉(zhuǎn)換為\u003c。

    ·數(shù)組和切片會轉(zhuǎn)換為JSON里面的數(shù)組,但[]btye類型的值將被轉(zhuǎn)換為Base64編碼后的字符串,slice類型額零值會被轉(zhuǎn)換為null。

    ·結(jié)構(gòu)體會轉(zhuǎn)換為JSON對象,并且只有結(jié)構(gòu)體里面以大寫字母開頭的可被導(dǎo)出的字段才會被轉(zhuǎn)換輸出,而這些可導(dǎo)出的字段會作為JSON對象的字符串索引。

    ·轉(zhuǎn)換一個map類型的數(shù)據(jù)結(jié)構(gòu)時,該數(shù)據(jù)的類型必須是map[string]T(T可以是encoding/json包支持的任意數(shù)據(jù)類型)。

2.解碼JSON數(shù)據(jù)

  可以使用json.Unmarshal()函數(shù)將JSON格式的文本解碼為Go里面預(yù)期的數(shù)據(jù)結(jié)構(gòu)。json.Unmarshal()函數(shù)的原型如下:

如何在Go語言中處理JSON

  該函數(shù)的第一次參數(shù)是輸入,即JSON格式的文本(比特序列),第二個參數(shù)表示目標(biāo)輸出容器,用于存放解碼后的值。

  要解碼一段JSON數(shù)據(jù),首先需要在Go中創(chuàng)建一個目標(biāo)類型的實例對象,用于存放解碼后的值:

      var book BOOK

  然后調(diào)用json.Unmarshal()函數(shù),將[]byte類型的JSON數(shù)據(jù)作為第一個參數(shù)傳入,將book實例變量的指針作為第二個參數(shù)傳入:

      err := json.Unmarshal(b, &book)

  如果b是一個有效的JSON數(shù)據(jù)并能和book結(jié)構(gòu)對應(yīng)起來,那么JSON解碼后的值將會一一存放到book結(jié)構(gòu)體中。解碼成功后的book數(shù)據(jù)如下:

如何在Go語言中處理JSON

    json.Unmarshal()函數(shù)會根據(jù)一個約定的順序查找目標(biāo)結(jié)構(gòu)中的字段,如果找到一個即發(fā)生匹配。假設(shè)一個JSON對象有個名為"Foo"的索引,要將"Foo"所對應(yīng)的值填充到目標(biāo)結(jié)構(gòu)體的目標(biāo)字段上,json.Unmarshal()將會遵循如下順序進行查找匹配:

        (1)一個包含F(xiàn)oo標(biāo)簽的字段;

        (2)一個名為Foo的字段;

        (3)一個名為Foo或者除了首字母其他字母不區(qū)分大小的命名為Foo的字段。

    這些字段在類型聲明中必須都是以大寫字母開頭、可被導(dǎo)出的字段。

    如果JSON中的字段在Go目標(biāo)類型中不存在,json.Unmarshal()函數(shù)在解碼過程中會丟棄該字段。

3.解碼未知結(jié)構(gòu)的JSON數(shù)據(jù)

 Go內(nèi)建了靈活的類型系統(tǒng),向我們傳達了一個很有價值的信息:空接口是通用類型。如果要解碼一段未知結(jié)構(gòu)的JSON,只需將這段JSON數(shù)據(jù)解碼輸出到一個空接口即可。在解碼JSON數(shù)據(jù)的過程中,JSON數(shù)據(jù)里面的元素類型將做如下轉(zhuǎn)換:

  ·JSON中的布爾值將會轉(zhuǎn)換為Go中的bool類型;

  ·數(shù)值會被轉(zhuǎn)換為Go中的float64類型;

  ·字符串轉(zhuǎn)換后還是string類型;

  ·JSON數(shù)組會轉(zhuǎn)換為[]interface{}類型;

  ·JSON對象會轉(zhuǎn)換為map[string]interface{}類型;

  ·null值會被轉(zhuǎn)換為nil.

   在Go的標(biāo)準(zhǔn)庫encoding/json包中,允許使用map[string] interface{}和[]interface{}類型的值來分別存放未知結(jié)構(gòu)的JSON對象或數(shù)組,實例代碼如下:

如何在Go語言中處理JSON

   在上述代碼中,r被定義為一個空接口。json.Unmarshal()函數(shù)將一個JSON對象解碼到空接口r中,最終r將會是一個鍵值對的map[string] interface{}結(jié)構(gòu):

 如何在Go語言中處理JSON 

   要訪問解碼后的數(shù)據(jù)結(jié)構(gòu),需要先判斷目標(biāo)結(jié)構(gòu)是否為預(yù)期的數(shù)據(jù)類型:

如何在Go語言中處理JSON

   然后,我們可以通過for循環(huán)搭配range語句一一訪問解碼后的目標(biāo)數(shù)據(jù):

如何在Go語言中處理JSON

4.JSON的流式讀寫

   Go內(nèi)建的encoding/json包還提供Decoder和Encoder兩個類型,用于支持JSON數(shù)據(jù)的流式,并提供NewDecoder()和NewEncoder()兩個函數(shù)來便于具體實現(xiàn):

如何在Go語言中處理JSON

    使用Decoder 和Encoder對數(shù)據(jù)流進行處理可以應(yīng)用得更為廣泛些,比如讀寫 HTTP 連接、WebSocket或文件等, Go的標(biāo)準(zhǔn)庫net/rpc/jsonrpc就是一個應(yīng)用了Decoder和Encoder的實際例子。

實例代碼:把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 
}

到此,關(guān)于“如何在Go語言中處理JSON”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識,請繼續(xù)關(guān)注億速云網(wǎng)站,小編會繼續(xù)努力為大家?guī)砀鄬嵱玫奈恼拢?/p>

向AI問一下細節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI