使用反射SetString

Dav*_*ide 4 string reflection struct go

我有一个像这样的结构:

type ProductionInfo struct {
    StructA []Entry
}

type Entry struct {
    Field1 string
    Field2 int
}
Run Code Online (Sandbox Code Playgroud)

我想更改Field1使用反射的值,但反射对象始终返回CanSet() = false。我能做些什么?请参阅游乐场示例。

https://play.golang.org/p/eM_KHC3kQ5

这是代码:

func SetField(source interface{}, fieldName string, fieldValue string) {
    v := reflect.ValueOf(source)
    tt := reflect.TypeOf(source)

    for k := 0; k < tt.NumField(); k++ {
        fieldValue := reflect.ValueOf(v.Field(k))

        fmt.Println(fieldValue.CanSet())
        if fieldValue.CanSet() {
            fieldValue.SetString(fieldValue.String())
        }
    }
}

func main() {
    source := ProductionInfo{}
    source.StructA = append(source.StructA, Entry{Field1: "A", Field2: 2})

    SetField(source, "Field1", "NEW_VALUE")
}
Run Code Online (Sandbox Code Playgroud)

icz*_*cza 6

多个错误。让我们迭代一下它们。

首先,您传递的是一个值,而不是要修改其字段ProductionInfo的值,因此首先将其更改为:Entry

SetField(source.StructA[0], "Field1", "NEW_VALUE")
Run Code Online (Sandbox Code Playgroud)

接下来,您将传递一个(非指针)值。您无法使用反射修改非指针结构的字段,因为这只会修改将被丢弃的副本。为了避免这种情况(以及进一步的混乱),这是不允许的(CanSet()返回false)。所以你必须传递一个指向结构的指针:

SetField(&source.StructA[0], "Field1", "NEW_VALUE")
Run Code Online (Sandbox Code Playgroud)

现在内部将SetField()描述reflect.ValueOf(source)传递的指针。您可以使用Value.Elem()导航到reflect.Value指向的对象(结构值):

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

现在它起作用了。修改后的代码:

func SetField(source interface{}, fieldName string, fieldValue string) {
    v := reflect.ValueOf(source).Elem()

    fmt.Println(v.FieldByName(fieldName).CanSet())

    if v.FieldByName(fieldName).CanSet() {
        v.FieldByName(fieldName).SetString(fieldValue)
    }
}

func main() {
    source := ProductionInfo{}
    source.StructA = append(source.StructA, Entry{Field1: "A", Field2: 2})

    fmt.Println("Before: ", source.StructA[0])
    SetField(&source.StructA[0], "Field1", "NEW_VALUE")
    fmt.Println("After: ", source.StructA[0])
}
Run Code Online (Sandbox Code Playgroud)

输出(在Go Playground上尝试):

Before:  {A 2}
true
After:  {NEW_VALUE 2}
Run Code Online (Sandbox Code Playgroud)