Go json 未调用 Unmarshaller

Go json Unmarshaller is not called

我对 Golang 还很陌生。我一直在寻找一种方法来为编组和解组做一些自定义的事情 json。我找到了实现 MarshallerUnmarshaller 接口的解决方案。

这是我的 struct 实现的接口(我也实现了 Stringer):

type Data struct {
    Foo string `json:"foo"`
    bar string
}

func (d Data) MarshalJSON() ([]byte, error) {
    return []byte("{\"foo\":\"test\",\"bar\":\"data\"}"), nil
}

func (d Data) String() string {
    return fmt.Sprintf("Foo: %s, bar: %s", d.Foo, d.bar)
}

func (d Data) UnmarshalJSON(b []byte) error {
    d.bar = "testtest"
    d.Foo = "data"
    return nil
}

对于 Marshaller 一切都按预期工作:

data := &Data{}
marshal, _ := json.Marshal(data)
fmt.Println(string(marshal))

预期的输出:

{"foo":"test","bar":"data"}

但是 Unmarshaller 没有像我预期的那样工作:

jsonData := "{\"foo\":\"test\"}"
data := Data{}
json.Unmarshal([]byte(jsonData), data)
fmt.Println(data)

此代码打印:

Foo: , bar:

我在这里遗漏了什么吗?

Data方法有值接收者,但变量数据有指针类型(*Data)。

指针类型变量不能使用带有值接收者的函数。更改变量类型或接收器类型。

另见 https://github.com/golang/go/wiki/CodeReviewComments#receiver-type 关于接收器。

这里有几个明显的问题。

  • 如果你想修改接收者,方法接收者必须是一个指针,否则你只是修改方法本地的副本。

  • 您总是需要解组为指针。

  • 您正在为 Foo 字段声明 "field" json 标记,但传入 "foo"

  • 您在 UnmarshalJSON 方法中调用 json.Unmarshal,该方法将无限期递归。

一个工作示例看起来像

func (d *Data) UnmarshalJSON(b []byte) error {
    type data Data
    tmp := &data{bar: "bar"}
    err := json.Unmarshal(b, tmp)
    if err != nil {
        return err
    }
    *d = Data(*tmp)
    return nil
}

func main() {
    jsonData := "{\"field\":\"test\"}"
    data := Data{}
    json.Unmarshal([]byte(jsonData), &data)
    fmt.Printf("%#v\n", data)
}

https://play.golang.org/p/FnF2li63dt