先通过reflect.MethodByName获取方法再调用,可实现动态方法调度。定义Calculator结构体及其Add、Multiply方法后,使用reflect.ValueOf获取实例反射值,调用MethodByName("Add")得到方法并传入[]reflect.Value参数,执行Call获得返回值,输出结果为15;还可通过Type.Method遍历所有导出方法,注意仅首字母大写的方法可见,需确保参数与接收者类型匹配。

在 Go 语言中,reflect.Method 可用于动态获取结构体的方法并进行调用。这在处理插件式逻辑、配置驱动调用或需要运行时动态行为的场景中非常有用。下面通过一个简单示例演示如何使用反射来调用结构体的方法。
定义结构体和方法
先定义一个包含多个方法的结构体,便于后续反射调用:
type Calculator struct{}
func (c *Calculator) Add(a, b int) int {
return a + b
}
func (c Calculator) Multiply(a, b int) int {
return a b
}
通过 reflect.Method 获取并调用方法
使用 reflect.Value.MethodByName 获取方法值,再通过 Call 动态调用:
package mainimport ( "fmt" "reflect" )
func main() { calc := &Calculator{} v := reflect.ValueOf(calc)
// 获取名为 "Add" 的方法 method := v.MethodByName("Add") if !method.IsValid() { fmt.Println("方法不存在") return } // 准备参数(必须是 reflect.Value 类型) args := []reflect.Value{ reflect.ValueOf(10), reflect.ValueOf(5), } // 调用方法 result := method.Call(args) // 获取返回值 fmt.Println("Add 结果:", result[0].Int()) // 输出: 15}
立即学习“go语言免费学习笔记(深入)”;
遍历所有可用方法
也可以通过 Type.Method(i) 遍历结构体的所有导出方法:
t := reflect.TypeOf(calc)
fmt.Printf("共有 %d 个方法\n", t.NumMethod())
for i := 0; i < t.NumMethod(); i++ {
method := t.Method(i)
fmt.Printf("方法名: %s\n", method.Name)
}
输出结果类似:
共有 2 个方法 方法名: Add 方法名: Multiply
注意:反射只能访问导出方法(首字母大写),非导出方法不会出现在 Method 列表中。
基本上就这些。掌握 MethodByName 和 Call 的组合使用,就能实现基本的动态方法调度。不复杂但容易忽略参数和接收者类型匹配的问题。










