Go 语言 reflect 包仅支持调用已导出函数或方法,需确保可寻址性、导出性及参数类型严格匹配;调用前须校验函数类型、参数数量与兼容性,推荐封装安全调用器而非依赖 panic 捕获。
Go 语言的 reflect 包支持在运行时检查类型、值,甚至调用函数——但要注意:它只能调用已导出(首字母大写)的函数或方法,且参数和返回值需严格匹配。动态调用不是“万能胶”,而是有明确边界的安全操作。
反射无法访问未导出(小写开头)的函数或方法。例如:
✅ 正确(可反射调用):func Add(a, b int) int { return a + b }type Calculator struct{}func (c Calculator) Multiply(x, y int) int { return x * y }
func subtract(a, b int) int { return a - b }(小写开头)func (c Calculator) divide(x, y int) int { return x / y }(未导出方法)
核心步骤:获取函数的 reflect.Value → 构造参数切片 → 调用 → 处理返回值。
示例:调用 Add(10, 20)
reflect.ValueOf(Add) 获取函数值[]reflect.Value 类型,每个元素用 reflect.ValueOf(arg) 包装[]reflect.Value,按顺序取结果(如 results[0].Int())方法不是独立函数,必须关联到具体实例(receiver):
reflect.ValueOf(&calc) 获取结构体指针的 Value.MethodByName("Multiply") 获取方法值(注意:传入的是指针,否则非指针接收者方法可能不可调用)x, y)⚠️ 若方法定义为 func (c Calculator) Foo(),则 c 必须是可寻址的(即传指针或地址);否则 MethodByName 返回无效值。
反射易 panic,务必检查关键状态:
fn.Ki
nd() == reflect.Func 确认是函数fn.Type().NumIn() == len(args) 校验参数个数fn.Type().In(i).AssignableTo(arg.Type()) 检查参数类型是否兼容(避免 runtime panic)defer func(){ if r := recover(); r != nil { /*日志*/ } }()
更推荐:封装一个带类型检查的通用调用器,或结合 code generation(如 go:generate)在编译期生成安全调用代码。
基本上就这些。反射调用不复杂但容易忽略 receiver 可寻址性、导出规则和类型匹配——把这三点盯牢,90% 的问题就解决了。