我刚接触Golang。我一直在寻找一种方法来做一些自定义的工作,以便封送和解码json。我已经找到实现MarshallerUnmarshaller接口(interface)的解决方案。

这是我的带有已实现接口(interface)的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))

预期的输出:



但是Unmarshaller无法正常运行:
jsonData := "{\"foo\":\"test\"}"
data := Data{}
json.Unmarshal([]byte(jsonData), data)
fmt.Println(data)

此代码打印:



我在这里缺少什么吗?

最佳答案

您在这里有一些明显的问题。

  • 如果要修改接收者,则方法接收者必须是一个指针,否则,您将只修改该方法本地的副本。
  • 您始终需要解码指针。
  • 您要为"field"字段声明一个Foo json标签,但要传递"foo"
  • 您正在json.Unmarshal方法中调用UnmarshalJSON,该方法将无限期递归。

  • 一个工作的例子看起来像
    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

    10-06 09:22