给出以下代码:

package main

import (
    "encoding/json"
    "log"
)

type Somefin string

func (s *Somefin) UnmarshalJSON(b []byte) error {
    log.Println("Unmarshaling",string(b))
    *s = Somefin("~"+string(b)+"~")
    return nil
}

type Wat struct {
    A, B string
    *Somefin
}

func main() {
    b := []byte(`{"A":"foo","B":"bar","Somefin":"baz"}`)
    w := &Wat{Somefin: new(Somefin)}

    err := json.Unmarshal(b,w)
    log.Println(w, err)
}

我得到以下输出:
# go run wat.go
2013/12/14 13:59:17 Unmarshaling {"A":"foo","B":"bar","Somefin":"baz"}
2013/12/14 13:59:17 &{  <nil>} <nil>

因此,出于某种原因,Somefin key 试图解码整个结构,而不是仅仅解码它应该的 key 。我做错了吗,或者这是json编码器中的错误?顺便说一句,正在进行1.2。

最佳答案

为什么最后没有结果

这不是解码器中的错误,而是代码中的错误。您只是分配另一个地址
s中的本地指针UnmarshalJSON。更正的代码:

func (s *Somefin) UnmarshalJSON(b []byte) error {
    log.Println("Unmarshaling",string(b))
    sn := Somefin("~"+string(b)+"~")
    *s = sn
    return nil
}
s = &sn的语义:将地址&sn分配给s。这类似于s = 42
*s = sn的语义:将sn的任何内容复制到s指向的位置。

要执行此操作的一项要求是s指向有效的内存位置,并且不能为nil
您的代码(play)的示例用法:
w := &Wat{Somefin: new(Somefin)}

err := json.Unmarshal(b,w)
log.Printf("%#v (%s)\n", w, err)

至关重要的是使用新的Wat初始化Somefin,以便指针sUnmarshalJSON有效(指向使用new(Somefin)创建的对象)。

为什么要在UnmarshalJSON中获取整个字符串

嵌入不是多态。而嵌入对象的方法集(在您的情况下Somefin)被提升到外部,这并不意味着该方法现在可以正常工作
在嵌入式结构而不是嵌入式结构上。

小例子(play):
type Inner struct { a int }
func (i *Inner) A() int { return i.a }

type Outer struct {
    *Inner
    a int
}

i := &Inner{}
o := Outer{Inner: i}

fmt.Println("Inner.A():", i.A())
fmt.Println("Outer.A():", o.A())

o.a = 2

fmt.Println("Outer.A():", o.A())

使用多态,您会期望Outer.A()返回2,因为A()方法将在Outer的范围,不再是Inner。嵌入后,范围永远不会改变,并且A()将始终在Inner上运行。

同样的效果可以防止UnmarshalJSON看到这两个成员AB,因为这两个成员
根本不在Somefin的范围内:
  • JSON库在UnmarshalJSON上看到Wat,因为UnmarshalJSON中的Somefin被提升为外部
  • JSON库无法在Somefin中找到任何匹配的元素,并交付整个输入
  • 10-07 23:49