欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時(shí)間:2023年03月14日 11:03:43   作者:NJU_lemon  
本文主要介紹了Go json反序列化“null“的問題解決,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

有這么一段代碼,可以先看一下有沒有什么問題,作用是輸入一段json字符串,反序列化成map,然后將另一個(gè)inputMap的內(nèi)容,merge進(jìn)這個(gè)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"會(huì)發(fā)生什么呢?

實(shí)驗(yàn)

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)
}

不出意外的話,要出意外了

它說我給一個(gè)nil map賦值了,但我明明在反序列化之前給jsonMap初始化了的,原來,jsoniter這個(gè)庫【其他庫沒測(cè)】在進(jìn)行json序列化時(shí),會(huì)把nil【即指針類型(比如slice、map和*T)的未初始化時(shí)的形態(tài)】序列化為字符串"null",反序列化時(shí)會(huì)把字符串"null",如果目標(biāo)類型是指針類型,則會(huì)反序列化為nil

其他測(cè)試

知道現(xiàn)象和原因之后,我又測(cè)試了些其他的東西
需要注意的是fmt很多時(shí)候打印nil的指針類型時(shí)不會(huì)輸出nil,比如nil的slice和map,會(huì)打印成[]和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), &params); 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

補(bǔ)充說明

默認(rèn)的反序列化是用float64來接數(shù)字類型的,原來的數(shù)字太大會(huì)出現(xiàn)精度丟失問題

"null"用int或float32等基礎(chǔ)數(shù)字類型來接會(huì)變成默認(rèn)值0

很多json庫在反序列化時(shí)都會(huì)存在精度丟失問題,比如int64的最后幾位變成0,是因?yàn)椴幻鞔_json字符串代表的struct的場景下,用map[string]interface{}來接反序列化之后的內(nèi)容,會(huì)默認(rèn)用float64來接數(shù)字類型,“int64是將64bit的數(shù)據(jù)全部用來存儲(chǔ)數(shù)據(jù),但是float64需要表達(dá)的信息更多,因此float64單純用于數(shù)據(jù)存儲(chǔ)的位數(shù)將小于64bit,這就導(dǎo)致了float64可存儲(chǔ)的最大整數(shù)是小于int64的。”,理論上數(shù)值超過9007199254740991就可能會(huì)出現(xiàn)精度缺失,反序列化時(shí)需要針對(duì)數(shù)字類型單獨(dú)處理【用struct來接,并且保證類型能對(duì)應(yīng)上就不會(huì)有以上問題】:

 到此這篇關(guān)于Go json反序列化“null“的問題解決的文章就介紹到這了,更多相關(guān)Go json反序列化內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Go外部依賴包從vendor,$GOPATH和$GOPATH/pkg/mod查找順序

    Go外部依賴包從vendor,$GOPATH和$GOPATH/pkg/mod查找順序

    這篇文章主要介紹了Go外部依賴包vendor,$GOPATH和$GOPATH/pkg/mod下查找順序,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • golang逐行讀取文件的操作

    golang逐行讀取文件的操作

    這篇文章主要介紹了golang逐行讀取文件的操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2020-12-12
  • Go框架三件套Gorm?Kitex?Hertz基本用法與常見API講解

    Go框架三件套Gorm?Kitex?Hertz基本用法與常見API講解

    這篇文章主要為大家介紹了Go框架三件套Gorm?Kitex?Hertz的基本用法與常見API講解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪<BR>
    2023-02-02
  • Golang中HTTP路由設(shè)計(jì)的使用與實(shí)現(xiàn)

    Golang中HTTP路由設(shè)計(jì)的使用與實(shí)現(xiàn)

    這篇文章主要介紹了Golang中HTTP路由設(shè)計(jì)的使用與實(shí)現(xiàn),為什么要設(shè)計(jì)路由規(guī)則,因?yàn)槁酚梢?guī)則是HTTP的請(qǐng)求按照一定的規(guī)則 ,匹配查找到對(duì)應(yīng)的控制器并傳遞執(zhí)行的邏輯,需要的朋友可以參考下
    2023-05-05
  • 詳解go如何使用xorm在執(zhí)行前改寫?SQL

    詳解go如何使用xorm在執(zhí)行前改寫?SQL

    這篇文章主要為大家介紹了詳解go如何使用xorm在執(zhí)行前改寫SQL的實(shí)現(xiàn)過程詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-06-06
  • Golang?PHP?數(shù)據(jù)綁定示例分析

    Golang?PHP?數(shù)據(jù)綁定示例分析

    這篇文章主要為大家介紹了Golang?PHP?數(shù)據(jù)綁定示例分析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-08-08
  • 深入理解Go語言設(shè)計(jì)模式之函數(shù)式選項(xiàng)模式

    深入理解Go語言設(shè)計(jì)模式之函數(shù)式選項(xiàng)模式

    在 Go 語言中,函數(shù)選項(xiàng)模式(Function Options Pattern)是一種常見且強(qiáng)大的設(shè)計(jì)模式,用于構(gòu)建可擴(kuò)展、易于使用和靈活的 API,本文就來看看它的具體用法吧
    2023-05-05
  • golang通過mysql語句實(shí)現(xiàn)分頁查詢

    golang通過mysql語句實(shí)現(xiàn)分頁查詢

    這篇文章主要介紹了golang通過mysql語句實(shí)現(xiàn)分頁查詢,文章內(nèi)容介紹詳細(xì),具有一定的參考價(jià)值,需要的小伙伴可以參考一下,希望對(duì)你的學(xué)習(xí)有所幫助
    2022-03-03
  • 如何使用Go語言獲取當(dāng)天、昨天、明天、某天0點(diǎn)時(shí)間戳以及格式化時(shí)間

    如何使用Go語言獲取當(dāng)天、昨天、明天、某天0點(diǎn)時(shí)間戳以及格式化時(shí)間

    這篇文章主要給大家介紹了關(guān)于如何使用Go語言獲取當(dāng)天、昨天、明天、某天0點(diǎn)時(shí)間戳以及格式化時(shí)間的相關(guān)資料,格式化時(shí)間戳是將時(shí)間戳轉(zhuǎn)換為特定的日期和時(shí)間格式,文中通過代碼示例介紹的非常詳細(xì),需要的朋友可以參考下
    2023-10-10
  • golang?中?recover()的使用方法

    golang?中?recover()的使用方法

    這篇文章主要介紹了Guam與golang??recover()的使用方法,Recover?是一個(gè)Go語言的內(nèi)建函數(shù),可以讓進(jìn)入宕機(jī)流程中的?goroutine?恢復(fù)過來,下文更多相關(guān)資料需要的小伙伴可以參考一下
    2022-04-04

最新評(píng)論