您好,登錄后才能下訂單哦!
本文小編為大家詳細介紹“Go json反序列化“null“的問題如何解決”,內容詳細,步驟清晰,細節處理妥當,希望這篇“Go json反序列化“null“的問題如何解決”文章能幫助大家解決疑惑,下面跟著小編的思路慢慢深入,一起來學習新知識吧。
有這么一段代碼,可以先看一下有沒有什么問題,作用是輸入一段json字符串,反序列化成map,然后將另一個inputMap的內容,merge進這個map
func mergeContent(inputJson string, inputMap map[string]interface{}) (map[string]interface{}, error) { jsonMap := make(map[string]interface{}) if inputJson != "" { decoder := jsoniter.NewDecoder(strings.NewReader(inputJson)) decoder.UseNumber() if err := decoder.Decode(&jsonMap); err != nil { return nil, err } } //merge for k, v := range inputMap { jsonMap[k] = v } return jsonMap, nil }
看上去是不是一段很健康的代碼?
結合標題再看看呢?
如果輸入的json字符串是"null"會發生什么呢?
func main(){ inputMap := make(map[string]interface{}) inputMap["test"] = 1 outputMap, err := mergeContent("null", inputMap) if err != nil { fmt.Println("err:", err) return } fmt.Printf("output:%+v\n", outputMap) }
不出意外的話,要出意外了
它說我給一個nil map賦值了,但我明明在反序列化之前給jsonMap初始化了的,原來,jsoniter這個庫【其他庫沒測】在進行json序列化
時,會把nil
【即指針類型(比如slice、map和*T)的未初始化時的形態】序列化為字符串"null",反序列化
時會把字符串"null"
,如果目標類型是指針類型,則會反序列化為nil
知道現象和原因之后,我又測試了些其他的東西
需要注意的是fmt很多時候打印nil的指針類型時不會輸出nil,比如nil的slice和map,會打印成[]和map[]
;
package main import ( "fmt" jsoniter "github.com/json-iterator/go" ) type Marshaler struct { A map[string]interface{} B []string C [10]int D *string E *EE F string g string } type EE struct { EEE []string } func main() { mal := Marshaler{ E: &EE{}, } mal1 := &Marshaler{} e1 := &EE{} e2 := EE{EEE: []string{}} var t1 *string var t2 []int var t3 map[string]interface{} var t4 = make([]string, 0) res1, _ := jsoniter.MarshalToString(mal) res2, _ := jsoniter.MarshalToString(mal1) res3, _ := jsoniter.MarshalToString(e1) res4, _ := jsoniter.MarshalToString(e2) res5, _ := jsoniter.MarshalToString(t1) res6, _ := jsoniter.MarshalToString(t2) res7, _ := jsoniter.MarshalToString(t3) res8, _ := jsoniter.MarshalToString(t4) fmt.Printf("res1: %+v\n", res1) fmt.Printf("res2: %+v\n", res2) fmt.Printf("res3: %+v\n", res3) fmt.Printf("res4: %+v\n", res4) fmt.Printf("res5: %+v\n", res5) fmt.Printf("res6: %+v\n", res6) fmt.Printf("res7: %+v\n", res7) fmt.Printf("res8: %+v\n", res8) params := make(map[string]interface{}) if err := jsoniter.Unmarshal([]byte(res6), ¶ms); err != nil { fmt.Println("null Unmarshal err:", err) } else { fmt.Printf("null Unmarshal map: %+v\n", params) } if err := jsoniter.Unmarshal([]byte(res6), &mal); err != nil { fmt.Println("null Unmarshal err:", err) } else { fmt.Printf("null Unmarshal Marshaler: %+v\n", mal) } if err := jsoniter.Unmarshal([]byte(res6), &mal1); err != nil { fmt.Println("null Unmarshal err:", err) } else { fmt.Printf("null Unmarshal Marshaler: %+v\n", mal1) } if err := jsoniter.Unmarshal([]byte(res6), &t4); err != nil { fmt.Println("null Unmarshal err:", err) } else { fmt.Printf("null Unmarshal []string: %+v\n", t4) fmt.Println(t4 == nil) } }
輸出:
res1: {"A":null,"B":null,"C":[0,0,0,0,0,0,0,0,0,0],"D":null,"E":{"EEE":null},"F"
:""}
res2: {"A":null,"B":null,"C":[0,0,0,0,0,0,0,0,0,0],"D":null,"E":null,"F":""}
res3: {"EEE":null}
res4: {"EEE":[]}
res5: null
res6: null
res7: null
res8: []
//下面的打印不夠準確,看debug截圖
null Unmarshal map: map[]
null Unmarshal Marshaler: {A:map[] B:[] C:[0 0 0 0 0 0 0 0 0 0] D:<nil> E:0xc000
004510 F: g:}
null Unmarshal Marshaler: <nil>
null Unmarshal []string: []
true
補充說明
默認的反序列化是用float64來接數字類型的,原來的數字太大會出現精度丟失問題
"null"用int或float32等基礎數字類型來接會變成默認值0
很多json庫在反序列化時都會存在精度丟失問題,比如int64的最后幾位變成0,是因為不明確json字符串代表的struct的場景下,用map[string]interface{}來接反序列化之后的內容,會默認用float64來接數字類型,“int64是將64bit的數據全部用來存儲數據,但是float64需要表達的信息更多,因此float64單純用于數據存儲的位數將小于64bit,這就導致了float64可存儲的最大整數是小于int64的。”,理論上數值超過9007199254740991就可能會出現精度缺失,反序列化時需要針對數字類型單獨處理【用struct來接,并且保證類型能對應上就不會有以上問題】:
讀到這里,這篇“Go json反序列化“null“的問題如何解決”文章已經介紹完畢,想要掌握這篇文章的知識點還需要大家自己動手實踐使用過才能領會,如果想了解更多相關內容的文章,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。