登录
首页 >  Golang >  Go问答

利用反射填充指向结构的指针

来源:stackoverflow

时间:2024-02-27 23:45:25 183浏览 收藏

大家好,我们又见面了啊~本文《利用反射填充指向结构的指针》的内容中将会涉及到等等。如果你正在学习Golang相关知识,欢迎关注我,以后会给大家带来更多Golang相关文章,希望我们能一起进步!下面就开始本文的正式内容~

问题内容

我想迭代结构中的字段并提示将字符串值输入到字符串字段,对指向结构的指针的字段递归地执行此操作。

目前这是我尝试过的方法,但在尝试在指针的字符串字段中设置此值时出现错误。

package main

import (
    "fmt"
    "reflect"
)

type Table struct {
    PK *Field
}

type Field struct {
    Name string
}


func main() {
    PopulateStruct(&Table{})
}

func PopulateStruct(a interface{}) interface {} {
    typeOf := reflect.TypeOf(a)
    valueOf := reflect.ValueOf(a)
    for i := 0; i < typeOf.Elem().NumField(); i++ {
        switch typeOf.Elem().Field(i).Type.Kind() {
        case reflect.String:
            fmt.Print(typeOf.Elem().Field(i).Name)
            var s string
            fmt.Scanf("%s", &s)
            valueOf.Elem().Field(i).SetString(s)
        case reflect.Ptr:
            ptr := reflect.New(valueOf.Elem().Field(i).Type())
            PopulateStruct(ptr.Elem().Interface())
            valueOf.Elem().Field(i).Set(ptr)
        }
    }
}

期望返回值包含带有指针字符串字段集的初始化结构。

设置指针的字符串字段时出现错误。

恐慌:反射:对零值调用reflect.value.field


解决方案


我将您的代码按原样放入 go 演示中,但它没有构建,因为 populatestruct 被声明为返回 interface{} 但实际上没有返回任何内容。删除声明的返回类型会产生您提到的恐慌。

这是因为在外部 populatestruct 调用的入口处,您有一个有效的指针,指向零值 table。零值 table 有一个元素:其中有一个 *field 类型的 nil 指针。因此,您的循环运行一次并找到 reflect.ptr,如您所料。添加更多诊断打印消息有助于了解发生的情况:

fmt.printf("populatestruct: i have typeof=%v, valueof=%v\n", typeof, valueof)
for i := 0; i < typeof.elem().numfield(); i++ {
    switch typeof.elem().field(i).type.kind() {
// ... snipped some code ...
    case reflect.ptr:
        ptr := reflect.new(valueof.elem().field(i).type())
        fmt.println("after allocating ptr, we have:", ptr.type(), ptr,
            "but its elem is:", ptr.elem().type(), ptr.elem())

打印:

populatestruct: i have typeof=*main.table, valueof=&{}
after allocating ptr, we have: **main.field 0x40c138 but its elem is: *main.field 

考虑到 populatestruct 本身的构造方式,我们必须在调用 populatestruct 之前现在实际分配一个真实的 field 实例。我们可以通过以下方式做到这一点:

p2 := ptr.Elem()
        ptr.Elem().Set(reflect.New(p2.Type().Elem()))

code borrowed from json.Unmarshal)。现在我们可以填写这个 field,它有一个名为 name 的字段,类型为 string

在我看来,这里的总体策略并不是那么好:填充可能应该采用通用指针,而不是专门指向 struct 指针的指针。然后,您可以在 json 解组器中模拟 indirect 函数。但是,添加这两行(创建目标对象并使分配的指针指向它)足以使现有代码运行。

(或者,您可以从头开始创建并返回整个实例,在这种情况下,您需要的只是类型,但我假设您有一个模式,其中只有一些字段无。)

Here's the complete Go Playground example. 我做了一些其他更改,因为使用演示时没有任何内容可供扫描。

到这里,我们也就讲完了《利用反射填充指向结构的指针》的内容了。个人认为,基础知识的学习和巩固,是为了更好的将其运用到项目中,欢迎关注golang学习网公众号,带你了解更多关于的知识点!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>