golang自定义json序列化应用

咖啡加方糖 · · 2793 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

问题引入

当某个struct存在某个字段为string或者[]byte类型但是实际上保存的内容是json格式的数据时,对其进行json序列化,比如

type Message struct {
    From string     `json:"from"`
    To   string     `json:"to"`
    Data string `json:"data"`
}

func main() {
    msg := Message{
        From: "XiaoMing",
        To:   "LiGang",
        Data: `{"title":"test","body":"something"}`,
    }
    jsonData, err := json.Marshal(msg)
    if err != nil {
        panic(err)
    }
    fmt.Println(string(jsonData))
}

在上面的例子中,Data字段是string类型,但是保存的内容是json格式的数据,这个时候,程序输出:

{"from":"XiaoMing","to":"LiGang","data":"{\"title\":\"test\",\"body\":\"something\"}"}

可以看到,序列化之后的data是一个字符串。
如果Message对应的是数据库中的一张表,而data字段在数据库中是json类型,当我们需要一个接口,查询Message表中的记录返回给客户端。如果直接执行序列化,那么客户端获取到的Data实际上是一个字符串,客户端还需要自行对这个字符串进行json反序列化。

这时候我们就会想,有没有什么办法能够在服务端序列化Message时,将data字段序列化成json对象而不是字符串呢?

自定义序列化

因为data字段的值本身就是json类型,为什么不能在序列化时直接使用呢?
查看json包的官方文档,我们可以发现关于 自定义序列化的例子
当执行json序列化时,如果对应的类型实现了Marshaler接口:

type Marshaler interface {
    MarshalJSON() ([]byte, error)
}

那么就会执行其MarshalJSON方法,并将返回的字节数组作为该值的序列化值。
那么回到上面的例子,我们就很容易实现目标:

type JsonString string

func (j JsonString) MarshalJSON() ([]byte, error) {
    fmt.Println("marshal...")
    return []byte(j), nil
}

type Message struct {
    From string     `json:"from"`
    To   string     `json:"to"`
    Data JsonString `json:"data"`
}

在上面的代码中基于string类型声明了JsonString,代表json格式的字符串,并实现了Marshaler接口。因为JsonString代表的就是json字符串,直接将其转换成字节数组返回。
然后将Message中的Data字段换成JsonString类型。
再次执行程序,可以看到:

{"from":"XiaoMing","to":"LiGang","data":{"title":"test","body":"something"}}

Perfect!


有疑问加站长微信联系(非本文作者)

本文来自:简书

感谢作者:咖啡加方糖

查看原文:golang自定义json序列化应用

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

2793 次点击  
加入收藏 微博
1 回复  |  直到 2018-08-14 09:30:24
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传