GO 中运行时类型的新结构实例

New instance of struct from the type at runtime in GO

我正在尝试创建结构的新实例,在运行时使用它的类型 (reflect.TypeOf)。我在 Whosebug How do you create a new instance of a struct from it's Type at runtime in Go?. Here is my implementation (also at http://play.golang.org/p/BtX0d5Ytu8):

上关注了这个话题
package main

import (
    "fmt"
    "reflect"
    "encoding/json"
    "bytes"
)

type UserInfo struct {
    Email    string `json:"email"`
    FullName string `json:"name"`
    ID       string `json:"_id"`
}

func main() {
    fmt.Println("Hello, playground")
    db := DBEngine{}
    db.DB = make(map[string][]byte)

    db.Register(UserInfo{})

    db.Put("142321", UserInfo{"jdoe@acme.com", "John Doe", "142321"})
    ret := db.Get("142321")
    fmt.Println("TypeOf(ret):", reflect.TypeOf(ret))
    fmt.Println("ValueOf(ret):", reflect.ValueOf(ret))
    fmt.Println("Value:", ret)
}

type DBEngine struct {
    Template interface{}
    DB       map[string][]byte
}

func (db *DBEngine) Register(v interface{}) {
    db.Template = v
}

//Set User defined object
func (db *DBEngine) Put(key string, v interface{}) {
    res, _ := json.Marshal(v)
    db.DB[key] = res
}

//Return user defined object
func (db *DBEngine) Get(key string) interface{} {
    decoder := json.NewDecoder(bytes.NewReader(db.DB[key]));
    fmt.Println("Value []byte:", string(db.DB[key]))
    ret := reflect.New(reflect.TypeOf(db.Template)).Elem()
    fmt.Println(reflect.TypeOf(db.Template), ret)
    decoder.Decode(ret)
    return ret.Interface()
}

出于某种原因,我总是得到空结构。我无法设置字段或修改。有人可以提出问题所在吗?

我查看了您的代码。当你用反射新建一个类型时,你得到了一个 Value 类型的值,需要调用 Interface() 来获取新生成的值的接口

检查代码 http://play.golang.org/p/CHWSV8EG7D

On 运行 reflect.New(sometype) 导致指向某种类型的 reflect.Value 的指针。如果您使用 .Elem() 将获得空的 reflect.Value 的直接值。为此 main.UserInfo{}。

//Return user defined object
func (db *DBEngine) Get(key string) interface{} {
    decoder := json.NewDecoder(bytes.NewReader(db.DB[key]));
    fmt.Println("Value []byte:", string(db.DB[key]))
    ret := reflect.New(reflect.TypeOf(db.Template)).Elem()//<--wrong
    fmt.Println(reflect.TypeOf(db.Template), ret)
    decoder.Decode(ret)
    return ret.Interface()
}
//print
//Hello, playground
//Value []byte: {"email":"jdoe@acme.com","name":"John Doe","_id":"142321"}
//main.UserInfo {  }
//TypeOf(ret): main.UserInfo
//ValueOf(ret): {  }
//Value: {  }

要获取 *sometype 值,请使用 .Interface(),这样您就可以解码了。参见:

//Return user defined object
func (db *DBEngine) Get(key string) interface{} {
    decoder := json.NewDecoder(bytes.NewReader(db.DB[key]));
    fmt.Println("Value []byte:", string(db.DB[key]))
    retValue := reflect.New(reflect.TypeOf(db.Template))
    retInter := retValue.Interface()//<-- 
    fmt.Println(reflect.TypeOf(db.Template), retInter)
    decoder.Decode(retInter)
    return retInter
}

结果:

Hello, playground
Value []byte: {"email":"jdoe@acme.com","name":"John Doe","_id":"142321"}
main.UserInfo &{  }
TypeOf(ret): *main.UserInfo
ValueOf(ret): &{jdoe@acme.com John Doe 142321}
Value: &{jdoe@acme.com John Doe 142321}

操场上:https://play.golang.org/p/veTUho5D4d