用 reflect.Kind 可区分基础类型(如 int、string)和复合类型(如 struct、slice、map),它只看底层形态,不关心命名;自定义类型 Kind 与其底层一致,需 Type() 获取名称和包路径。
reflect.Kind 区分基础类型和复合类型reflect.Kind 表示底层类型的分类,比如 int、struct、slice,它不关心具体是哪个命名类型(如 type UserID int),只看“长得像什么”。这是做动态分支判断最常用的依据。
int、string、bool 等,Kind() 返回对应基础种类,不是 reflect.Int 就是 reflect.String
type Status string),Kind() 仍是 reflect.String,但 Type().Name() 才是 "Status"
Kind() 返回 reflect.Ptr、reflect.Slice、reflect.Map,而不是其元素类型val := reflect.ValueOf([]int{1, 2})
fmt.Println(val.Kind()) // slice
fmt.Println(val.Type().Elem()) // int
reflect.Type 而不是 reflect.Kind
当你需要识别「命名类型」或做类型精确匹配时,必须用 Type。比如序列化时保留结构体字段名、校验是否为某个特定 type MyError struct{}、或实现泛型替代逻辑。
Type.Name() 只对具名类型(包级定义)返回非空字符串;匿名结构体、函数、内联切片返回空Type.PkgPath() 判断是否来自当前包,避免误匹配同名但不同包的类型t1 == t2(reflect.Type 是可比较的接口),别用 == 比较 Kind
var u UserID t := reflect.TypeOf(u) fmt.Println(t.Name(), t.PkgPath()) // "UserID" "your/package"
reflect.Value.Interface() 常 panic,怎么安全调用这个方法只对可导出(首字母大写)字段、非空接口值、以及未被 reflect.ValueOf(nil) 初始化的值才安全。最常见 panic 是 “call of reflect
.Value.Interface on zero Value” 或 “cannot interface with unexported field”。
v.IsValid() 排除零值(比如 reflect.Value{} 或 nil 指针解引用)v.CanInterface() 确保能转成 interface{}(例如结构体字段未导出时返回 false)v.Interface(),应先 v.Elem() 再判有效性v := reflect.ValueOf(&struct{ name string }{"alice"})
if v.IsValid() && v.CanInterface() {
fmt.Println(v.Interface()) // panic: unexported field
}
// 正确做法:解引用后取字段需单独检查
v = v.Elem().FieldByName("name")
if v.IsValid() && v.CanInterface() {
fmt.Println(v.Interface()) // 不会 panic
}
Kind 判断容易漏掉哪一层结构体字段可能是指针、接口或嵌套结构体,Kind() 只反映当前层级,不会自动展开。比如 interface{} 存了 *User,它的 Kind 是 reflect.Interface,不是 reflect.Ptr —— 必须先 v.Elem() 或 v.Interface() 后再重新反射。
reflect.Interface,先用 v.Elem() 获取内部值,再检查是否有效reflect.Ptr,记得 v.Elem() 前先 v.IsValid() && !v.IsNil()
Kind,要逐层 Elem() / Interface() / Index() 后再重新取 Kind
最常被忽略的是:接口值本身是 reflect.Interface,但它装的东西可能是任意 Kind,这层跳转必须手动做,reflect 不会替你“穿透”。