reflect.TypeOf 返回接口变量的静态类型,需传入接口所持具体值才能获取真实类型;reflect.ValueOf 可获取底层值但需确保可导出和可寻址;Interface() 方法 panic 常因字段未导出或值不可导出;推荐用 Kind() 和 Name()+PkgPath() 安全判断类型。
Go 的接口变量本身只存类型信息和数据指针,reflect.TypeOf 返回的是接口的静态类型(即接口类型本身),不是底层具体类型。想看到真实类型,必须先解包——也就是传入接口值的底层数据,而不是接口变量本身。
常见错误是直接对接口变量调用:
var i interface{} = "hello"
fmt.Println(reflect.TypeOf(i)) // 输出:interface {},不是 string这输出的是接口类型,没用。
nil 接口,reflect.TypeOf 会返回 nil,需提前判空reflect.TypeOf 对指针、切片、map 等复合类型返回的是完整描述,比如 *string、[]int

reflect.ValueOf 返回的是运行时值的封装,它能反映接口背后的原始值,但前提是该值可寻址或可导出。对未导出字段或不可寻址的临时值(如字面量、函数返回值),部分操作会 panic。
典型误用:
var i interface{} = struct{ name string }{"alice"}
v := reflect.ValueOf(i)
fmt.Println(v.Field(0)) // panic: cannot access unexported field
v.CanAddr() 为 false,此时不能调用 Addr() 或修改字段v.Type().Name()(仅对命名类型有效),否则用 v.Type().String()
reflect.Value.Interface() 用于把反射值转回 interface{},但它有严格前提:该 Value 必须是可导出的(即对应字段/变量是大写开头),且不能是零值或未初始化状态。
常见 panic 场景:
type T struct{ x int }
v := reflect.ValueOf(T{}).Field(0) // x 是小写字段
v.Interface() // panic: reflect: call of reflect.Value.Interface on unexported field
Field(0) 返回的 Value 就不可调用 Interface()
Index() 后得到的 Value 默认不可导出,也不能直接 Interface()
reflect.Value.Addr().Interface() 前提是原值可取地址且字段导出别依赖 reflect.TypeOf(x).String() 做类型分支——字符串匹配脆弱且不可靠。应结合 reflect.Kind 和 reflect.Type 的比较。
func typeCheck(v interface{}) {
rv := reflect.ValueOf(v)
rt := rv.Type()
switch rt.Kind() {
case reflect.String:
fmt.Println("string")
case reflect.Struct:
if rt.Name() == "Time" && rt.PkgPath() == "time" {
fmt.Println("time.Time")
}
case reflect.Ptr:
if rt.Elem().Name() == "Buffer" && rt.Elem().PkgPath() == "bytes" {
fmt.Println("*bytes.Buffer")
}
}
}
Kind() 是基础分类(如 struct、ptr、slice),稳定可靠Name() + PkgPath() 组合才能唯一标识一个命名类型,避免同名冲突Name() 为空,只能靠 String() 或字段结构推断真正难的不是拿到类型,而是决定要不要反射——多数场景用类型断言(v, ok := i.(string))更安全、更快。反射只在类型完全未知且必须动态处理时才值得引入。