Go json反序列化“null“的問題解決
有這么一段代碼,可以先看一下有沒有什么問題,作用是輸入一段json字符串,反序列化成map,然后將另一個inputMap的內(nèi)容,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
}
看上去是不是一段很健康的代碼?
結(jié)合標(biāo)題再看看呢?
如果輸入的json字符串是"null"會發(fā)生什么呢?
實驗
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)的未初始化時的形態(tài)】序列化為字符串"null",反序列化時會把字符串"null",如果目標(biāo)類型是指針類型,則會反序列化為nil
其他測試
知道現(xiàn)象和原因之后,我又測試了些其他的東西
需要注意的是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: []
//下面的打印不夠準(zhǔn)確,看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來接數(shù)字類型的,原來的數(shù)字太大會出現(xiàn)精度丟失問題
"null"用int或float32等基礎(chǔ)數(shù)字類型來接會變成默認值0
很多json庫在反序列化時都會存在精度丟失問題,比如int64的最后幾位變成0,是因為不明確json字符串代表的struct的場景下,用map[string]interface{}來接反序列化之后的內(nèi)容,會默認用float64來接數(shù)字類型,“int64是將64bit的數(shù)據(jù)全部用來存儲數(shù)據(jù),但是float64需要表達的信息更多,因此float64單純用于數(shù)據(jù)存儲的位數(shù)將小于64bit,這就導(dǎo)致了float64可存儲的最大整數(shù)是小于int64的。”,理論上數(shù)值超過9007199254740991就可能會出現(xiàn)精度缺失,反序列化時需要針對數(shù)字類型單獨處理【用struct來接,并且保證類型能對應(yīng)上就不會有以上問題】:

到此這篇關(guān)于Go json反序列化“null“的問題解決的文章就介紹到這了,更多相關(guān)Go json反序列化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
golang 并發(fā)安全Map以及分段鎖的實現(xiàn)方法
這篇文章主要介紹了golang 并發(fā)安全Map以及分段鎖的實現(xiàn)方法,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-03-03
go中利用reflect實現(xiàn)json序列化的示例代碼
和Java語言一樣,Go也實現(xiàn)運行時反射,這為我們提供一種可以在運行時操作任意類型對象的能力,本文給大家介紹了在go中如何利用reflect實現(xiàn)json序列化,需要的朋友可以參考下2024-03-03
一文詳解Go語言fmt標(biāo)準(zhǔn)庫的常用占位符使用
這篇文章主要為大家詳細介紹了Go語言中fmt標(biāo)準(zhǔn)庫的常用占位符及其簡單使用,文中的示例代碼講解詳細,對我們學(xué)習(xí)Go語言有一定的幫助,需要的可以參考一下2022-12-12
Go打印結(jié)構(gòu)體提升代碼調(diào)試效率實例詳解
這篇文章主要介紹了Go打印結(jié)構(gòu)體提升代碼調(diào)試效率實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2024-02-02
GoRoutines高性能同時進行多個Api調(diào)用實現(xiàn)
這篇文章主要為大家介紹了GoRoutines高性能同時進行多個Api調(diào)用實現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-03-03
golang架構(gòu)設(shè)計開閉原則手寫實現(xiàn)
這篇文章主要為大家介紹了golang架構(gòu)設(shè)計開閉原則手寫實例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-07-07

