首頁 > 軟體

Go json反序列化「null「的問題解決

2023-03-15 06:01:41

有這麼一段程式碼,可以先看一下有沒有什麼問題,作用是輸入一段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:%+vn", 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: %+vn", res1)
   fmt.Printf("res2: %+vn", res2)
   fmt.Printf("res3: %+vn", res3)
   fmt.Printf("res4: %+vn", res4)
   fmt.Printf("res5: %+vn", res5)
   fmt.Printf("res6: %+vn", res6)
   fmt.Printf("res7: %+vn", res7)
   fmt.Printf("res8: %+vn", res8)

   params := make(map[string]interface{})
   if err := jsoniter.Unmarshal([]byte(res6), &params); err != nil {
      fmt.Println("null Unmarshal err:", err)
   } else {
      fmt.Printf("null Unmarshal map:  %+vn", params)
   }

   if err := jsoniter.Unmarshal([]byte(res6), &mal); err != nil {
      fmt.Println("null Unmarshal err:", err)
   } else {
      fmt.Printf("null Unmarshal Marshaler: %+vn", mal)
   }

   if err := jsoniter.Unmarshal([]byte(res6), &mal1); err != nil {
      fmt.Println("null Unmarshal err:", err)
   } else {
      fmt.Printf("null Unmarshal Marshaler: %+vn", mal1)
   }

   if err := jsoniter.Unmarshal([]byte(res6), &t4); err != nil {
      fmt.Println("null Unmarshal err:", err)
   } else {
      fmt.Printf("null Unmarshal []string: %+vn", 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“的問題解決的文章就介紹到這了,更多相關Go json反序列化內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com