Go json 未调用 Unmarshaller
Go json Unmarshaller is not called
我对 Golang 还很陌生。我一直在寻找一种方法来为编组和解组做一些自定义的事情 json
。我找到了实现 Marshaller
和 Unmarshaller
接口的解决方案。
这是我的 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)
}
我对 Golang 还很陌生。我一直在寻找一种方法来为编组和解组做一些自定义的事情 json
。我找到了实现 Marshaller
和 Unmarshaller
接口的解决方案。
这是我的 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)
}