json.Unmarshal 带有后期类型断言的接口指针

json.Unmarshal interface pointer with later type assertion

因为我经常解组 http.Response.Body,我想我可以编写一个函数来处理读取、关闭和解组到各种不同结构的所有麻烦。这就是为什么我引入了一个函数 func unmarhalInterface(closer *io.ReadCloser, v *interface{}) error 然后可以用 t:=i.(T).

断言 return 值

根据 ,我已经将它包装成 *interface{} 类型的值,但是因为覆盖类型是 interface{} 而不是 myStruct,所以 json 包实现选择 map[string]interface{}。之后类型断言失败(当然)。有什么我遗漏的或需要此实现类型断言 "by hand",这意味着查找地图中的所有字段并将我想要的字段分配到我的结构中。

下面的代码在注释中有带符号的最小示例。如果我的解释不够充分,请走开。

package main

import (
    "bytes"
    "encoding/json"
    "fmt"
    "io"
    "io/ioutil"
    "log"
)

type myStruct struct {
    A string `json:"a"`
    B string `json:"b"`
}

func main() {
    jsonBlob := []byte(`{"a":"test","b":"test2"}`)

    var foo = interface{}(myStruct{})
    closer := ioutil.NopCloser(bytes.NewReader(jsonBlob))

    err := unmarshalCloser(&closer, &foo)
    if err != nil {
        log.Fatal(err)
    }
    fmt.Println(fmt.Sprintf("%v", foo))

    // That´s what i want:
    foo2 := foo.(myStruct)
    fmt.Println(foo2.A)
}

func unmarshalCloser(closer *io.ReadCloser, v *interface{}) error {
    defer func() { _ = (*closer).Close() }()

    data, err := ioutil.ReadAll(*closer)
    if err != nil {
        return err
    }

    err = json.Unmarshal(data, v)
    if err != nil {
        return err
    }
    return nil
}

Golang Playground

空接口不是实际类型,它基本上是可以匹配任何东西的东西。如评论中所述,指向空接口的指针实际上没有意义,因为指针已经匹配空接口,因为所有内容都匹配空接口。为了让你的代码工作,你应该删除你的结构周围的接口包装器,因为那只会搞乱 json 类型检查,而空接口的全部意义在于你可以传递 任何东西 到它。

package main

import (
    "bytes"
    "encoding/json"
    "fmt"
    "io"
    "io/ioutil"
    "log"
)

type myStruct struct {
    A string `json:"a"`
    B string `json:"b"`
}

func main() {
    jsonBlob := []byte(`{"a":"test","b":"test2"}`)

    var foo = &myStruct{} // This need to be a pointer so its attributes can be assigned
    closer := ioutil.NopCloser(bytes.NewReader(jsonBlob))

    err := unmarshalCloser(closer, foo)
    if err != nil {
        log.Fatal(err)
    }
    fmt.Println(fmt.Sprintf("%v", foo))

    // That´s what i want:
    fmt.Println(foo.A)
}

// You don't need to declare either of these arguments as pointers since they're both interfaces
func unmarshalCloser(closer io.ReadCloser, v interface{}) error {
    defer closer.Close()
    // v NEEDS to be a pointer or the json stuff will barf

    // Simplified with the decoder
    return json.NewDecoder(closer).Decode(v)
}