如何控制Go編碼JSON數(shù)據(jù)時的行為(問題及解決方案)
今天來聊一下我在Go中對數(shù)據(jù)進(jìn)行 JSON 編碼時遇到次數(shù)最多的三個問題以及解決方法,大家來看看是不是也為這些問題撓掉了不少頭發(fā)。
自定義JSON鍵名
這個問題加到文章里我是有所猶豫的,因為基本上大家都會,不過屬于同類問題我還是放進(jìn)來了,對新接觸 Go
的同學(xué)更友好些。
我們先從最常見的一個問題說,首先在Go 程序中要將數(shù)據(jù)編碼成JSON 格式時通常我們會先定義結(jié)構(gòu)體類型,將數(shù)據(jù)存放到結(jié)構(gòu)體變量中。
type Address struct { Type string City string Country string } type CreditCard struct { FirstName string LastName string Addresses []*Address Remark string } home := &Address{"private", "Aartselaar", "Belgium"} office := &Address{"work", "Boom", "Belgium"} card := VCard{"Jan", "Kersschot", []*Address{home, office}, "none"} js, err := json.Marshal(card) fmt.Printf("JSON format: %s", js)
只有導(dǎo)出的結(jié)構(gòu)體成員才會被編碼,這也就是我們?yōu)槭裁催x擇用大寫字母開頭的字段名稱。在編碼時,默認(rèn)使用結(jié)構(gòu)體字段的名字作為JSON對象中的 key
,但是一般JSON 是給 HTTP接口
返回數(shù)據(jù)使用的,在接口的規(guī)范里針對數(shù)據(jù)我們一般都要求返回 snake case
風(fēng)格的字段名。解決這個問題的方法是在結(jié)構(gòu)體聲明時在結(jié)構(gòu)體字段標(biāo)簽里可以自定義對應(yīng)的 JSON key
所以我們把結(jié)構(gòu)體聲明改為如下即可:
type Address struct { Type string `json:"type"` City string `json:"city"` Country string `json:"country"` }
編碼JSON時忽略掉指定字段
并不是所有數(shù)據(jù)我們都期望編碼到 JSON
中暴露給外部接口的,所以針對一些敏感的字段我們往往希望將其從編碼后的 JSON
數(shù)據(jù)中忽略掉。那么上面也說了只有導(dǎo)出的結(jié)構(gòu)體成員才會被編碼,有的同學(xué)會問我直接用小寫的字段名不行嗎?可是未導(dǎo)出字段只能在包內(nèi)訪問,像這種攜帶內(nèi)部敏感數(shù)據(jù)的往往都是應(yīng)用的基礎(chǔ)數(shù)據(jù),由項目的公共包來提供的。那么怎么既能維持字段的導(dǎo)出性又能讓其在 JSON
數(shù)據(jù)中被忽略掉呢? 還是使用結(jié)構(gòu)體的標(biāo)簽進(jìn)行注解,比如下面定義的結(jié)構(gòu)體,可以把身份證 IdCard
字段在 JSON
數(shù)據(jù)中去掉:
type User struct { Name string `json:"name"` Age Int `json:"int"` IdCard string `json:"-"` }
encoding/json 的源碼中和文檔中都列舉了通過結(jié)構(gòu)體字段標(biāo)簽控制數(shù)據(jù) JSON 編碼行為的說明:
// Field is ignored by this package. Field int `json:"-"` // Field appears in JSON as key "myName". Field int `json:"myName"` // Field appears in JSON as key "myName" and // the field is omitted from the object if its value is empty, // as defined above. Field int `json:"myName,omitempty"` // Field appears in JSON as key "Field" (the default), but // the field is skipped if empty. // Note the leading comma. Field int `json:",omitempty"`
omitempty 這個是字段的數(shù)據(jù)為空時,在 JSON 中省略這個字段。為的是節(jié)省數(shù)據(jù)空間, Protobuf 編譯器生成的結(jié)構(gòu)體代碼中每個字段標(biāo)簽中都有 omitempty 。但是在 Api 開發(fā)中這個不常用,因為字段不固定對前端很不友好。
對 Protobuf 不了解的可以看我之前寫的文章《 Protobuf語言指南 》。
結(jié)構(gòu)體字段標(biāo)簽的 json 注解中都不加 omitempty 后還遇到一種情況,就是數(shù)據(jù)類型為切片的字段在數(shù)據(jù)為空的時候會被 JSON 編碼為 null 而不是 [] 。這個前端經(jīng)常會問我沒數(shù)據(jù)的時候能不能不要返回 null ,每回還要多寫一個判斷。我的說辭都是不能,其實規(guī)范點講是應(yīng)該返回 [] 的知識我是我自己沒找到到解決方法。作為一個在寫代碼上有強迫癥的人,這個問題還是想搞明白的,好在有一天在 StackOverflow 上看到一個答案,才發(fā)現(xiàn)是編碼的疏忽導(dǎo)致的。
解決空切片在JSON里被編碼成null
因為切片的零值為 nil ,無指向內(nèi)存的地址,所以當(dāng)以這種形式定義 var f []int 初始化 slice 后,在JSON中將其編碼為 null ,如果想在 JSON 中將空 slice 編碼為 [] 則需用make初始化 slice為其分配內(nèi)存地址:
運行下面的例子可以看出兩點的區(qū)別:
package main import ( "encoding/json" "fmt" ) type Person struct { Friends []string } func main() { var f1 []string f2 := make([]string, 0) json1, _ := json.Marshal(Person{f1}) json2, _ := json.Marshal(Person{f2}) fmt.Printf("%s\n", json1) fmt.Printf("%s\n", json2) }
輸出:
{"Friends":null } {"Friends":[] }
其實導(dǎo)致這個問題的原因是Go的 append 函數(shù)(甩鍋),我們都知道引用類型的變量定義后如果沒初始化他們的值是 nil ,無指向內(nèi)存的地址,是無法直接使用的。但是 append 函數(shù)在給切片追加元素時會判斷切片是否已初始化,沒有的話會幫其初始化分配底層數(shù)組。我的習(xí)慣是先聲明切片,然后再在下面的循環(huán)代碼中向切片追加元素。但是如果循環(huán)沒有執(zhí)行,比如你從數(shù)據(jù)庫沒查出數(shù)據(jù),就會導(dǎo)致對應(yīng)切片字段在無數(shù)據(jù)時返回的是 nil 然后被 JSON 編碼成了 null 。所以這個算是一個經(jīng)驗總結(jié)出來的 Tip 吧在寫代碼時大家一定要注意了。
這就是我在開發(fā)時把數(shù)據(jù)編碼成 JSON 格式時遇到的三個問題和相應(yīng)的解決方法。加上之前寫的 解析JSON的文章 ,兩個文章加起來差不多就能匯總?cè)粘i_發(fā)中關(guān)于 encoding/json 庫使用的各種問題了。
總結(jié)
以上所述是小編給大家介紹的如何控制Go編碼JSON數(shù)據(jù)時的行為,希望對大家有所幫助!
相關(guān)文章
GO語言中創(chuàng)建切片的三種實現(xiàn)方式
這篇文章主要介紹了GO語言中創(chuàng)建切片的三種實現(xiàn)方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-09-09使用Gin框架返回JSON、XML和HTML數(shù)據(jù)
Gin是一個高性能的Go語言Web框架,它不僅提供了簡潔的API,還支持快速的路由和中間件處理,在Web開發(fā)中,返回JSON、XML和HTML數(shù)據(jù)是非常常見的需求,本文將介紹如何使用Gin框架來返回這三種類型的數(shù)據(jù),需要的朋友可以參考下2024-08-08Go語言使用Timeout Context取消任務(wù)的實現(xiàn)
本文主要介紹了Go語言使用Timeout Context取消任務(wù)的實現(xiàn),包括基本的任務(wù)取消和控制HTTP客戶端請求的超時,具有一定的參考價值,感興趣的可以了解一下2024-01-01go標(biāo)準(zhǔn)庫net/http服務(wù)端的實現(xiàn)示例
go的http標(biāo)準(zhǔn)庫非常強大,本文主要介紹了go標(biāo)準(zhǔn)庫net/http服務(wù)端,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2024-07-07Go-客戶信息關(guān)系系統(tǒng)的實現(xiàn)
這篇文章主要介紹了Go-客戶信息關(guān)系系統(tǒng)的實現(xiàn),本文章內(nèi)容詳細(xì),具有很好的參考價值,希望對大家有所幫助,需要的朋友可以參考下2023-01-01