从界面获取所有字段

RAF*_*FJR 9 reflection struct go reply

我如何知道我可以从reply对象/接口访问的字段?我尝试过反射,但似乎你必须首先知道字段名称.如果我需要知道可用的所有字段怎么办?

// Do sends a command to the server and returns the received reply.
Do(commandName string, args ...interface{}) (reply interface{}, err error)
Run Code Online (Sandbox Code Playgroud)

icz*_*cza 29

您可以使用该reflect.TypeOf()函数来获取reflect.Type类型描述符.从那里,您可以列出存储在界面中的动态值的字段.

例:

type Point struct {
    X int
    Y int
}

var reply interface{} = Point{1, 2}
t := reflect.TypeOf(reply)
for i := 0; i < t.NumField(); i++ {
    fmt.Printf("%+v\n", t.Field(i))
}
Run Code Online (Sandbox Code Playgroud)

输出:

{Name:X PkgPath: Type:int Tag: Offset:0 Index:[0] Anonymous:false}
{Name:Y PkgPath: Type:int Tag: Offset:4 Index:[1] Anonymous:false}
Run Code Online (Sandbox Code Playgroud)

Type.Field()调用的结果是一个reflect.StructField值,它struct包含字段的名称以及其他内容:

type StructField struct {
    // Name is the field name.
    Name string
    // ...
}
Run Code Online (Sandbox Code Playgroud)

如果您还想要字段的值,您可以使用reflect.ValueOf()获取reflect.Value(),然后您可以使用Value.Field()Value.FieldByName():

v := reflect.ValueOf(reply)
for i := 0; i < v.NumField(); i++ {
    fmt.Println(v.Field(i))
}
Run Code Online (Sandbox Code Playgroud)

输出:

1
2
Run Code Online (Sandbox Code Playgroud)

Go Playground尝试一下.

注意:通常指向struct的指针包含在接口中.在这种情况下,您可以使用Type.Elem()Value.Elem()"导航"到指向的类型或值:

t := reflect.TypeOf(reply).Elem()

v := reflect.ValueOf(reply).Elem()
Run Code Online (Sandbox Code Playgroud)

如果您不知道它是否是一个指针或没有,你可以检查它Type.Kind()Value.Kind(),结果与比较reflect.Ptr:

t := reflect.TypeOf(reply)
if t.Kind() == reflect.Ptr {
    t = t.Elem()
}

// ...

v := reflect.ValueOf(reply)
if v.Kind() == reflect.Ptr {
    v = v.Elem()
}
Run Code Online (Sandbox Code Playgroud)

Go Playground上试试这个变种.

有关Go的反思的详细介绍,请阅读博客文章:反思的法则.