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)
多个错误。让我们迭代一下它们。
首先,您传递的是一个值,而不是要修改其字段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)