

[译]自定义Go Json的序列化方法
source link: https://colobu.com/2020/03/19/Custom-JSON-Marshalling-in-Go/
Go to the source link to view the article. You can view the picture content, updated content and better typesetting reading experience. If the link is broken, please click the button below to view the snapshot at that time.

编译自 Custom JSON Marshalling in Go 。
我们知道,通过tag,可以有条件地实现定制Go JSON序列化的方式,比如 json:",omitempty"
, 当字段的值为空的时候,我们可以在序列化后的数据中不包含这个值,而 json:"-"
可以直接不被JSON序列化,如果想被序列化key -
,可以设置tag为 json:"-,"
,加个逗号。
如果你为类型实现了 MarshalJSON() ([]byte, error)
和 UnmarshalJSON(b []byte) error
方法,那么这个类型在序列化反序列化时将采用你定制的方法。
这些都是我们常用的设置技巧。
如果临时想为一个struct增加一个字段的话,可以采用本译文的技巧,临时创建一个类型,通过嵌入原类型的方式来实现。他和 JSON and struct composition in Go
一文中介绍的技巧还不一样(译文和jsoniter-go扩展可以阅读陶文的 Golang 中使用 JSON 的一些小技巧
)。 JSON and struct composition in Go
一文中是通过嵌入的方式创建一个新的类型,你序列化和反序列化的时候需要使用这个新类型,而本译文中的方法是无痛改变原类型的 MarshalJSON
方式,采用 Alias
方式避免递归解析,确实是一种非常巧妙的方法。
以下是译文:
Go的 encoding/json
序列化 strcut
到JSON数据:
package main import ( "encoding/json" "os" "time" ) type MyUser struct { ID int64 `json:"id"` Name string `json:"name"` LastSeen time.Time `json:"lastSeen"` } func main() { _ = json.NewEncoder(os.Stdout).Encode( &MyUser{1, "Ken", time.Now()}, ) }
序列化的结果:
{"id":1,"name":"Ken","lastSeen":"2009-11-10T23:00:00Z"}
但是如果我们想改变一个字段的显示结果我们要怎么做呢?例如,我们想把 LastSeen
显示为unix时间戳。
最简单的方式是引入另外一个辅助struct,在 MarshalJSON
中使用它进行正确的格式化:
func (u *MyUser) MarshalJSON() ([]byte, error) { return json.Marshal(&struct { ID int64 `json:"id"` Name string `json:"name"` LastSeen int64 `json:"lastSeen"` }{ ID: u.ID, Name: u.Name, LastSeen: u.LastSeen.Unix(), }) }
这样做当然没有问题,但是如果有很多字段的话就会很麻烦,如果我们能把原始struct嵌入到新的struct中,并让它继承所有不需要改变的字段就太好了:
func (u *MyUser) MarshalJSON() ([]byte, error) { return json.Marshal(&struct { LastSeen int64 `json:"lastSeen"` *MyUser }{ LastSeen: u.LastSeen.Unix(), MyUser: u, }) }
但是等等,问题是这个辅助struct也会继承原始struct的 MarshalJSON
方法,这会导致这个方法进入无限循环中,最后堆栈溢出。
解决办法就是为原始类型起一个别名,别名会有原始struct所有的字段,但是不会继承它的方法:
func (u *MyUser) MarshalJSON() ([]byte, error) { type Alias MyUser return json.Marshal(&struct { LastSeen int64 `json:"lastSeen"` *Alias }{ LastSeen: u.LastSeen.Unix(), Alias: (*Alias)(u), }) }
同样的技术也可以应用于 UnmarshalJSON
方法:
func (u *MyUser) UnmarshalJSON(data []byte) error { type Alias MyUser aux := &struct { LastSeen int64 `json:"lastSeen"` *Alias }{ Alias: (*Alias)(u), } if err := json.Unmarshal(data, &aux); err != nil { return err } u.LastSeen = time.Unix(aux.LastSeen,0) return nil }
Recommend
-
137
404 页面不存在 404. 抱歉,您访问的资源不存在。 可能是网址有误,或者对应的内容被删除,或者处于私有状态。 代码改变世界,联系邮箱 [email protected]
-
158
MSON,让JSON序列化更快2018年01月09日 作者: 秦喆 芝任 天洲 赵鹏 文章链接 6426字 13分...
-
79
先来看一段 golang package main import ( "encoding/json" "fmt" ) func main() { data := map[string]string{ "str0": "Hello, world", "str1": "<", "str2"...
-
72
-
42
序列化和反序列化 1、JSON的序列化 1.1序列化 struct、map、slice 对于json的序列化和反序列化,go的encoding/json 包提供了一些列的方法。 常用的比如 func Marshal(v interface{}) ([]by...
-
37
Json序列化 指,将具有key-value(键 -> 值)结构的数据类型,例如:go语言中的map, slice,struct...序列化成json格式的字符串的操作。json是一种主流的数据传输格式,灵活轻便... 需要导入一个包 "encoding/json"
-
41
在Python的世界里,将一个对象以json格式进行序列化或反序列化一直是一个问题。Python标准库里面提供了json序列化的工具,我们可以简单的用 json.dumps 来将一个对象序列化。但是这种序列化仅支持python内置的基本类型。
-
6
dotnet C# 如何让 Json 序列化数组时序列化继承类的属性如果我使用的是具体的数组而我的数组是基类数组,而我传入子类的元素进行 json 序列化,可能发现 Json.NET 序列化没有包含子类元素的属性。如果要包含子类的属性或字段,可以在序列化的类数组定义为...
-
8
【Python】pickle&json序列化 2017年10月06日 Author: Guofei 文章归类: Python语法 ,文章编号: 1211 版权声明:本文作者是郭飞。转载随意...
-
4
在 Go 中,对于自定义结构的序列化和反序列化存在几个问题。 Q1:如何保证待反序列化的字符串只包含所定义的结构中的字段? 从 Go 1.10 起,标准库 encoding/j...
About Joyk
Aggregate valuable and interesting links.
Joyk means Joy of geeK