golang MarshalJson的實現(xiàn)
在 Go 語言中,MarshalJSON 是一個接口方法,允許自定義類型在進行 JSON 編碼時提供自定義的序列化邏輯。通過實現(xiàn) MarshalJSON 方法,你可以控制結(jié)構(gòu)體或其他類型在轉(zhuǎn)換為 JSON 時的表現(xiàn)。
基本用法
當你想要自定義某個類型的 JSON 表現(xiàn)時,可以實現(xiàn) json.Marshaler 接口,該接口只包含一個方法 MarshalJSON。實現(xiàn)該方法后,使用 json.Marshal 函數(shù)時會自動調(diào)用你定義的 MarshalJSON 方法。
示例
以下是一個示例,展示如何自定義結(jié)構(gòu)體的 JSON 序列化:
package main
import (
"encoding/json"
"fmt"
)
// 定義一個結(jié)構(gòu)體
type User struct {
Name string
Age int
Email string
}
// 為 User 實現(xiàn) MarshalJSON 方法
func (u User) MarshalJSON() ([]byte, error) {
// 自定義 JSON 輸出格式
return json.Marshal(struct {
FullName string `json:"name"`
Age int `json:"age"`
Email string `json:"email_address"`
}{
FullName: u.Name,
Age: u.Age,
Email: u.Email,
})
}
func main() {
user := User{Name: "Alice", Age: 30, Email: "alice@example.com"}
// Marshal 用戶對象為 JSON
jsonData, err := json.Marshal(user)
if err != nil {
fmt.Println("Error marshaling to JSON:", err)
return
}
fmt.Println(string(jsonData)) // 輸出: {"name":"Alice","age":30,"email_address":"alice@example.com"}
}
解釋
- 定義結(jié)構(gòu)體:我們定義了一個
User結(jié)構(gòu)體,包含Name、Age和Email字段。 - 實現(xiàn)
MarshalJSON方法:我們?yōu)?nbsp;User結(jié)構(gòu)體實現(xiàn)了MarshalJSON方法。在這個方法中,我們自定義了 JSON 輸出格式。 - 自定義輸出:在
MarshalJSON方法中,我們使用匿名結(jié)構(gòu)體來定義最終的 JSON 格式。 - 使用
json.Marshal:在main函數(shù)中,我們創(chuàng)建了一個User實例,并使用json.Marshal將其轉(zhuǎn)換為 JSON 字符串。
注意事項
- 錯誤處理:
MarshalJSON方法應返回error,以便在序列化過程中可以處理潛在的錯誤。 - 遞歸調(diào)用:在
MarshalJSON方法中,如果調(diào)用json.Marshal,需要確保所處理的結(jié)構(gòu)體不會遞歸調(diào)用自身的MarshalJSON方法。
Marshal函數(shù)將會遞歸遍歷整個對象,依次按成員類型對這個對象進行編碼。
類型轉(zhuǎn)換規(guī)則如下:
- bool類型:轉(zhuǎn)換為JSON的Boolean
- 整數(shù)、浮點數(shù)等數(shù)值類型: 轉(zhuǎn)換為JSON的Number
- string類型: 轉(zhuǎn)換為JSON的字符串(帶""引號)
- struct類型:轉(zhuǎn)換為JSON的Object,再根據(jù)各個成員的類型遞歸打包
- 數(shù)組或切片類型: 轉(zhuǎn)換為JSON的Array
- []byte類型: 會先進行base64編碼然后轉(zhuǎn)換為JSON字符串
- map類型:轉(zhuǎn)換為JSON的Object,key必須是string
- interface{}類型: 按照內(nèi)部的實際類型進行轉(zhuǎn)換
- nil類型: 轉(zhuǎn)為JSON的null
- channel,func等類型: 會返回UnsupportedTypeError
從golang到j(luò)son:
| golang | json |
|---|---|
| bool | Boolean |
| int、float等數(shù)字 | Number |
| string | String |
| []byte(base64編碼) | String |
| struct | Object,再遞歸打包 |
| array/slice | Array |
| map | Object |
| interface{} | 按實際類型轉(zhuǎn)換 |
| nil | null |
| channel,func | UnsupportedTypeError |
從json到golang:
| json | golang |
|---|---|
| Boolean | bool |
| Number | float64 |
| String | string |
| Array | []interface{} |
| Object | map[string]interface{} |
| null | nil |
避坑
json.marshal使用不當,會存在base64編碼問題問題出現(xiàn)在:[]byte 在json.marshal時會進行base64 encoding處理
解決辦法:使用json.RawMessagejson.RawMessage其實就是[]byte類型的重定義。可以進行強制類型轉(zhuǎn)換。
現(xiàn)在有這么一種場景,結(jié)構(gòu)體中的其中一個字段的格式是未知的:
type Command struct {
ID int
Cmd string
Args *json.RawMessage
}
使用json.RawMessage的話,Args字段在Unmarshal時不會被解析,直接將字節(jié)數(shù)據(jù)賦值給Args。我們可以能先解包第一層的JSON數(shù)據(jù),然后根據(jù)Cmd的值,再確定Args的具體類型進行第二次Unmarshal。
注意:一定要使用指針類型
*json.RawMessage,否則在Args會被認為是[]byte類型,在打包時會被打包成base64編碼的字符串。
使用interface{}, interface{}類型在Unmarshal時,會自動將JSON轉(zhuǎn)換為對應的數(shù)據(jù)類型:
JSON的boolean: 轉(zhuǎn)換為bool
JSON的數(shù)值: 轉(zhuǎn)換為float64
JSON的字符串: 轉(zhuǎn)換為string
JSON的Array: 轉(zhuǎn)換為[]interface{}
JSON的Object: 轉(zhuǎn)換為map[string]interface{}
JSON的null: 轉(zhuǎn)換為nil
需要注意的有兩個:
- 一是所有的JSON數(shù)值自動轉(zhuǎn)換為float64類型,使用時需要再手動轉(zhuǎn)換為需要的int,int64等類型。
- 二是JSON的object自動轉(zhuǎn)換為
map[string]interface{}類型,訪問時直接用JSON Object的字段名作為key進行訪問。在不知道JSON數(shù)據(jù)的格式時,可以使用interface{}。
自定義類型:如果希望自己定義對象的打包解包方式,可以實現(xiàn)以下的接口:
type Marshaler interface {
MarshalJSON() ([]byte, error)
}
type Unmarshaler interface {
UnmarshalJSON([]byte) error
}
實現(xiàn)該接口的對象需要將自己的數(shù)據(jù)打包和解包。如果實現(xiàn)了該接口,json在打包解包時則會調(diào)用自定義的方法,不再對該對象進行其他處理。
總結(jié)
通過實現(xiàn) MarshalJSON 方法,可以為 Go 的自定義類型提供靈活的 JSON 序列化控制。這使得在與 JSON 數(shù)據(jù)交互時,可以更好地控制數(shù)據(jù)的結(jié)構(gòu)和格式。
到此這篇關(guān)于golang MarshalJson的實現(xiàn)的文章就介紹到這了,更多相關(guān)golang MarshalJson內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Go?tablewriter庫提升命令行輸出專業(yè)度實例詳解
命令行工具大家都用過,如果是運維人員可能會編寫命令行工具來完成各種任務,命令行輸出的美觀和易讀性往往容易被忽視,很爛的輸出會讓人感覺不專業(yè),本文將介紹Go語言中牛逼的實戰(zhàn)工具tablewriter庫,使你在命令行輸出中展現(xiàn)出專業(yè)的一面2023-11-11
golang gopm get -g -v 無法獲取第三方庫的解決方案
這篇文章主要介紹了golang gopm get -g -v 無法獲取第三方庫的解決方案,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-05-05
使用golang引入外部包的三種方式:go get, go module, ve
這篇文章主要介紹了使用golang引入外部包的三種方式:go get, go module, vendor目錄,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-01-01

