Golang reflect.Method动态方法调用示例

先通过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 main

import ( "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

}

遍历所有可用方法

也可以通过 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 的组合使用,就能实现基本的动态方法调度。不复杂但容易忽略参数和接收者类型匹配的问题。