Go反射调用出错多为运行时panic,排查需严格验证每步前提:方法存在性、可寻址性、导出性、接口实现及返回值数量类型,缺一不可。
Go反射调用出错,往往不是编译报错,而是运行时 panic 或逻辑异常。排查关键在于:**不假设类
型安全,每一步都验证;不跳过可寻址性、导出性、方法存在性等前提条件**。
调用 MethodByName 后必须立即检查返回值是否有效:
sfunc := stype.MethodByName("Func1") 后加 if !sfunc.IsValid() { t.Fatal("method not found") }
(*MyStruct).Save),而你传的是值类型 MyStruct{},MethodByName 会返回无效值IsValid() 为 false反射调用后返回的 []reflect.Value 是黑盒,不能默认索引 0 是结果、1 是 error:
if len(ret)
errVal := ret[1]; if !errVal.Type().Implements(reflect.TypeOf((*error)(nil)).Elem().Type()) { t.Fatal("second return value is not error") }
ret[1].Interface().(error) —— 若类型不匹配会 panic,应先用 errVal.CanInterface() 和类型断言安全判断向 Call 传参时,每个 reflect.Value 都要合规:
reflect.ValueOf(x).Field(i) 会 panicCall 会失败reflect.ValueOf(&arg).Elem() 确保传的是指针所指的值,而非指针本身(除非方法签名明确要 *T)反射错误常在深层调用中爆发,靠堆栈难定位:
defer func(){ if r:=recover();r!=nil{ log.Printf("panic in %s: %v", fname, r) } }()
runtime.Caller(1) 打印触发反射的具体行号SetInt、Interface())单独包裹 recover,缩小问题范围基本上就这些。反射不是“写完就能跑”,而是“每步都要问:它现在是什么类型?能不能做这事?”——养成验证习惯,比事后 debug 快得多。