我的问题与这里的这个问题有关:
基本上它声称下面的表达式为真,如果someX是reflect.Value包含指针的 a
reflect.Indirect(reflect.ValueOf(someX)) === reflect.ValueOf(someX).Elem()
Run Code Online (Sandbox Code Playgroud)
如果是这样,那么为什么我下面的代码会在最后一行崩溃?
package main
import (
"reflect"
"log"
)
type Person struct {
Name string
}
func main() {
newitem := reflect.New(reflect.ValueOf(Person{}).Type())
log.Println(reflect.TypeOf(newitem)) // shows reflect.Value
log.Println(newitem.Type().Kind()) // shows it is a ptr
log.Println(reflect.Indirect(reflect.ValueOf(newitem))) // this line does not cause panic
log.Println(reflect.ValueOf(newitem).Elem()) // this line causes panic
}
Run Code Online (Sandbox Code Playgroud)
我一直很难理解 Go 中的反射包,而且很可能我误解了 Go 语言的一些基本方面,如我过去一周一直在问的堆栈溢出问题中所指出的那样。
让我们分解以下行:
log.Println(reflect.ValueOf(newitem).Elem())
Run Code Online (Sandbox Code Playgroud)
该值newItem是一个reflect.Value。该表达式reflect.ValueOf(newItem)返回一个reflect.Value包含 a reflect.Value。因为包含的值不是指针或接口,所以调用Elem()恐慌。
以下行不会恐慌,因为如果参数不是指针类型,reflect.Indirect 会返回其参数。
log.Println(reflect.Indirect(reflect.ValueOf(newitem)))
Run Code Online (Sandbox Code Playgroud)
问题是应用程序正在用reflect.Values 包装reflect.Values。直接使用reflect.Value,如下面的代码:
log.Println(reflect.Indirect(newitem))
log.Println(newitem.Elem())
Run Code Online (Sandbox Code Playgroud)