Go中不能直接用reflect.Method调用结构体方法,因其仅为元信息;须通过reflect.Value.MethodByName()获取可调用的reflect.Value后调用Call(),且仅支持导出方法和可寻址接收者。
在 Go 中,不能直接用 reflect.Method 调用结构体方法,因为 reflect.Method 只是方法的**元信息描述**(如名字、类型),不是可调用的函数值。真正能调用的是通过 reflect.Value.Method() 或 reflect.Value.MethodByName() 获取的 reflect.Value,它封装了可执行的方法对象。
Go 的反射只能调用**导出(大写开头)的方法**,且接收者必须是可寻址的(即不能是对不可寻址临时值的指针)。
reflect.ValueOf(&v) 获取结构体指针的 reflect.Value
.MethodByName("MethodName") 或 .Method(i) 得到方法对应的 reflect.Value
.Call([]reflect.Value{...}) 传参调用(参数需为 []reflect.Value 类型)示例:
type Person struct {
Name string
}
func (p *Person) Greet(msg string) string {
return "Hello, " + msg + "! I'm " + p.Name
}
p := Person{Name: "Alice"}
v := reflect.ValueOf(&p) // 必须取地址,否则无法调用指针接收者方法
method := v.MethodByName("Greet")
if method.IsValid() {
result := method.Call([]reflect.Value{
reflect.ValueOf("world"),
})
fmt.Println(result[0].String()) // "Hello, world! I'm Alice"
}
reflect.Method 是结构体类型的方法列表项(只读描述),用于遍历、检查方法签名;MethodByName 才返回可调用的 reflect.Value。
reflect.TypeOf(t).Method(i) 查看第 i 个导出方法名、类型、是否导出等reflect.ValueOf(&t).MethodByName(name) 获取可调用对象func (t T) M()),ValueOf(t) 也可调用;但若接收者是指针(func (t *T) M()),必须传 &t
所有参数和返回值都需转为 reflect.Value,且类型必须严格匹配。
reflect.ValueOf(x) 包装参数(支持基本类型、结构体、指针等)[]reflect.Value,按声明顺序对应每个返回值.Interface() 恢复原始类型(需类型断言)或 .Int()/.String() 等取值func() (int, error)),结果数组长度就是 2反射调用容易 panic,需提前校验。
method.IsValid() 和 method.Kind() == reflect.Func
Call 会 panicMethodByName 返回无效值var p *Person; reflect.ValueOf(p).MethodByName("Greet").Call(...) 会 panic)