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

go json轉(zhuǎn)換實踐中遇到的坑

 更新時間:2018年12月24日 11:54:15   作者:simpleapples  
在使用 go 語言開發(fā)過程中,經(jīng)常需要使用到 json 包來進(jìn)行 json 和 struct 的互相轉(zhuǎn)換,這篇文章主要介紹了go json轉(zhuǎn)換實踐中遇到的坑,非常具有實用價值,需要的朋友可以參考下

在使用 go 語言開發(fā)過程中,經(jīng)常需要使用到 json 包來進(jìn)行 json 和 struct 的互相轉(zhuǎn)換,在使用過程中,遇到了一些需要額外注意的地方,記錄如下。

整數(shù)變浮點數(shù)問題

假設(shè)有一個 Person 結(jié)構(gòu),其中包含 Age int64 和 Weight float64 兩個字段,現(xiàn)在通過 json 包將 Person 結(jié)構(gòu)轉(zhuǎn)為 map[string]interface{},代碼如下。

type Person struct {
 Name string
 Age int64
 Weight float64
}

func main() {
 person := Person{
  Name: "Wang Wu",
  Age: 30,
  Weight: 150.07,
 }

 jsonBytes, _ := json.Marshal(person)
 fmt.Println(string(jsonBytes))

 var personFromJSON interface{}
 json.Unmarshal(jsonBytes, &personFromJSON)

 r := personFromJSON.(map[string]interface{})
}

代碼執(zhí)行到這里看上去一切正常,但是打印一下 map[string]interface{} 就會發(fā)現(xiàn)不太對了。

fmt.Println(reflect.TypeOf(r["Age"]).Name()) // float64
fmt.Println(reflect.TypeOf(r["Weight"]).Name()) // float64

轉(zhuǎn)換成 map[string]interface{} 之后,原先的 uint64 和 float64 類型都被轉(zhuǎn)換成了 float64 類型,這顯然是不符合我們的預(yù)期的。

查看 json 的規(guī)范可以看到,在 json 中是沒有整型和浮點型之分的,所以現(xiàn)在可以理解 json 包中的 Unmarshal 方法轉(zhuǎn)出的數(shù)字類型為什么都是 float64 了,因為根據(jù) json 規(guī)范,數(shù)字都是同一種類型,那么對應(yīng)到 go 的類型中最接近的就是 float64 了。

json 包還針對這個問題提供了更好的解決方案,不過需要使用 json.Decoder 來代替 json.Unmarshal 方法,將 json.Unmarhsal 替換如下。

var personFromJSON interface{}

decoder := json.NewDecoder(bytes.NewReader(jsonBytes))
decoder.UseNumber()
decoder.Decode(&personFromJSON)

r := personFromJSON.(map[string]interface{})

這種方法首先創(chuàng)建了一個 jsonDecoder,然后調(diào)用了 UseNumber 方法,從文檔中可以知道,使用 UseNumber 方法后,json 包會將數(shù)字轉(zhuǎn)換成一個內(nèi)置的 Number 類型(而不是 float64),這個 Number 類型提供了轉(zhuǎn)換為 int64、float64 等多個方法。

 

時間格式

對于 json 格式,是沒有時間類型的,日期和時間以 json 格式存儲時,需要轉(zhuǎn)換為字符串類型。這就帶來了一個問題,日期時間的字符串表示有多種多樣,go 的 json 包支持的是哪一種呢?

使用下面的代碼來輸出 json.Marshal 方法將 Time 類型轉(zhuǎn)換為字符串后的格式。

type Person struct {
 Name string
 Birth time.Time
}

func main() {
 person := Person{
  Name: "Wang Wu",
  Birth: time.Now(),
 }

 jsonBytes, _ := json.Marshal(person)
 fmt.Println(string(jsonBytes)) // {"Name":"Wang Wu","Birth":"2018-12-20T16:22:02.00287617+08:00"}
}

根據(jù)輸出可以判斷,go 的 json 包使用的是 RFC3339 標(biāo)準(zhǔn)中定義的格式。接下來測試一下 json.Unmarshal 方法所支持的日期時間格式。

dateStr := "2018-10-12"

var person Person
jsonStr := fmt.Sprintf("{\"name\":\"Wang Wu\", \"Birth\": \"%s\"}", dateStr)
json.Unmarshal([]byte(jsonStr), &person)

fmt.Println(person.Birth) // 0001-01-01 00:00:00 +0000 UTC

對于形如 2018-10-12 的字符串,json 包并沒有成功將其解析,接下來我們把 time 包中支持的所有格式都試一下。

經(jīng)過試驗,發(fā)現(xiàn) json.Unmarshal 方法只支持 RFC3339 和 RFC3339Nano 兩種格式的轉(zhuǎn)換。還有一個需要注意的地方,使用 time.Now() 生成的時間是帶有一個 Monotonic Time 的,經(jīng)過 json.Marshal 轉(zhuǎn)換時候,由于 RFC3339 規(guī)范里沒有存放 Monotonic Time 的位置,會丟掉這一部分。

對于字段為空的處理

json 包對于空值的處理是一個非常容易出錯的地方,看下面代碼。

type Person struct {
 Name  string
 Age  int64
 Birth time.Time
 Children []Person
}

func main() {
 person := Person{}

 jsonBytes, _ := json.Marshal(person)
 fmt.Println(string(jsonBytes)) // {"Name":"","Age":0,"Birth":"0001-01-01T00:00:00Z","Children":null}
}

當(dāng) struct 中的字段沒有值時,使用 json.Marshal 方法并不會自動忽略這些字段,而是根據(jù)字段的類型輸出了他們的默認(rèn)空值,這往往和我們的預(yù)期不一致,json 包提供了對字段的控制手段,我們可以為字段增加 omitempty tag,這個 tag 會在字段值為零值(int 和 float 類型零值是 0,string 類型零值是 "",對象類型零值是 nil)時,忽略該字段。

type PersonAllowEmpty struct {
 Name  string    `json:",omitempty"`
 Age  int64    `json:",omitempty"`
 Birth time.Time   `json:",omitempty"`
 Children []PersonAllowEmpty `json:",omitempty"`
}

func main() {
 person := PersonAllowEmpty{}
 jsonBytes, _ := json.Marshal(person)
 fmt.Println(string(jsonBytes)) // {"Birth":"0001-01-01T00:00:00Z"}
}

可以看到,這次輸出的 json 中只有 Birth 字段了,string、int、對象類型的字段,都因為沒有賦值,默認(rèn)是零值,所以被忽略,對于日期時間類型,由于不可以設(shè)置為零值,也就是 0000-00-00 00:00:00,不會被忽略。

需要注意這樣的情況:如果一個人的年齡是 0 (對于剛出生的嬰兒,這個值是合理的),剛好是 int 字段的零值,在添加 omitempty tag 的情況下,年齡字段會被忽略。

如果想要某一個字段在任何情況下都被 json 包忽略,需要使用如下的寫法。

type Person struct {
 Name  string `json:"-"`
 Age  int64 `json:"-"`
 Birth time.Time `json:"-"`
 Children []string `json:"-"`
}

func main() {
 birth, _ := time.Parse(time.RFC3339, "1988-12-02T15:04:27+08:00")
 person := Person{
  Name: "Wang Wu",
  Age: 30,
  Birth: birth,
  Children: []string{},
 }

 jsonBytes, _ := json.Marshal(person)
 fmt.Println(string(jsonBytes)) // {}
}

可以看到,使用 json:"-" 標(biāo)簽的字段都被忽略了。

補充:golang string轉(zhuǎn)json的一些坑

先看一段代碼,起作用是把字符串轉(zhuǎn)換為結(jié)構(gòu)體對應(yīng)的json

type people struct {

 name string `json:"name"`

 age int `json:"age"`

 id int `json:"id"`

}

 

type student struct {

 people

 id int `json:"sid"`

}

 

func main() {

 msg := "{\"name\":\"zhangsan\", \"age\":18, \"id\":122463, \"sid\":122464}"

 var someOne student

 if err := json.Unmarshal([]byte(msg), &someOne); err == nil {

  fmt.Println(someOne)

  fmt.Println(someOne.people)

 } else {

  fmt.Println(err)

 }

}

仔細(xì)看看,有沒有錯?我只能說,這樣是輸出不出來答案的,賦值錯誤,看下面的運行結(jié)果:

傷腦筋啊,我仔細(xì)看了半天,發(fā)現(xiàn)在定義的people和student兩個結(jié)構(gòu)體下邊有綠色的波浪線(我用的vscode),像下邊這樣:

鼠標(biāo)放上去顯示的是:

大家都知道,golang中變量聲明成大寫和小寫能引用的范圍是不一樣的,那我就想了,大小寫問題???一臉懵逼把變量名首字母改成了大寫,然后...就行了,代碼變成了下邊這樣:

type people struct {

 Name string `json:"name"`

 Age int `json:"age"`

 ID int `json:"id"`

}

 

type student struct {

 people

 ID int `json:"sid"`

}

 

func main() {

 msg := "{\"name\":\"zhangsan\", \"age\":18, \"id\":122463, \"sid\":122464}"

 var someOne student

 if err := json.Unmarshal([]byte(msg), &someOne); err == nil {

  fmt.Println(someOne)

  fmt.Println(someOne.people)

 } else {

  fmt.Println(err)

 }

} 

輸出的結(jié)果這樣:

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • Go語言學(xué)習(xí)之context包的用法詳解

    Go語言學(xué)習(xí)之context包的用法詳解

    日常Go開發(fā)中,Context包是用的最多的一個了,幾乎所有函數(shù)的第一個參數(shù)都是ctx,那么我們?yōu)槭裁匆獋鬟fContext呢,Context又有哪些用法,底層實現(xiàn)是如何呢?相信你也一定會有探索的欲望,那么就跟著本篇文章,一起來學(xué)習(xí)吧
    2022-10-10
  • Go語言實現(xiàn)服務(wù)端消息接收和發(fā)送

    Go語言實現(xiàn)服務(wù)端消息接收和發(fā)送

    這篇文章主要為大家詳細(xì)介紹了Go語言實現(xiàn)服務(wù)端消息接收和發(fā)送功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-07-07
  • golang操作rocketmq的示例代碼

    golang操作rocketmq的示例代碼

    這篇文章主要介紹了golang操作rocketmq的示例代碼,代碼簡單易懂,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-04-04
  • Go學(xué)習(xí)筆記之map的聲明和初始化

    Go學(xué)習(xí)筆記之map的聲明和初始化

    map底層是由哈希表實現(xiàn)的,Go使用鏈地址法來解決鍵沖突,下面這篇文章主要給大家介紹了關(guān)于Go學(xué)習(xí)筆記之map的聲明和初始化的相關(guān)資料,需要的朋友可以參考下
    2022-11-11
  • 利用systemd部署golang項目的實現(xiàn)方法

    利用systemd部署golang項目的實現(xiàn)方法

    這篇文章主要介紹了利用systemd部署golang項目的實現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-11-11
  • Go語言利用ssh連接服務(wù)器的方法步驟

    Go語言利用ssh連接服務(wù)器的方法步驟

    這篇文章主要介紹了Go語言利用ssh連接服務(wù)器的方法步驟,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法

    golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法

    今天小編就為大家分享一篇golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-07-07
  • 淺析Go語言中Channel的各種用法

    淺析Go語言中Channel的各種用法

    這篇文章主要帶大家一起來學(xué)習(xí)一下Go語言中的if語句,也就是大家口中的判斷語句。文中的示例代碼講解詳細(xì),對我們學(xué)習(xí)Go語言有一定幫助,需要的可以參考一下
    2022-11-11
  • Go 語言單例模式示例詳解

    Go 語言單例模式示例詳解

    這篇文章主要為大家介紹了Go 語言單例模式示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-10-10
  • golang控制結(jié)構(gòu)select機(jī)制及使用示例詳解

    golang控制結(jié)構(gòu)select機(jī)制及使用示例詳解

    這篇文章主要介紹了golang控制結(jié)構(gòu)select機(jī)制及使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-10-10

最新評論