go中类型判断需分三层:先用kind()粗粒度分类,再用Name()和PkgPath()识别命名类型,最后用Elem()等展开嵌套类型;注意接口nil值会panic,应优先用ValueOf安全探查。

在 Go 中,reflect 包是运行时获取类型和值信息的核心工具。判断变量类型不是靠“是什么”,而是靠“能做什么”——即通过 reflect.typeof() 获取类型描述,再用其方法做分类判断。
获取基础类型信息:TypeOf 和 ValueOf
reflect.TypeOf() 返回 reflect.Type,描述类型的静态结构;reflect.ValueOf() 返回 reflect.Value,承载运行时的值与操作能力。两者常配合使用,但类型判断主要依赖 Type。
- 对任意变量调用
reflect.TypeOf(v),得到其编译期类型(不含接口动态值) - 若变量是接口类型(如
interface{}),TypeOf返回的是接口中实际值的底层类型,不是接口本身 - 注意:nil 接口传入
TypeOf会 panic,需先判空或用ValueOf的Kind()安全探查
用 Kind() 做第一层粗粒度分类
reflect.Kind 是 Go 类型系统的底层分类,共 27 种(如 Int、String、Struct、Ptr、Interface 等)。它比具体类型名更稳定,适合通用逻辑分支。
-
t := reflect.TypeOf(x); t.Kind()可直接判断是否为切片、映射、指针等复合类型 - 例如:
t.Kind() == reflect.Slice能同时匹配[]int、[]string、[]User - 注意:
Kind()不区分具体命名类型,type MyInt int的Kind仍是Int,不是MyInt
用 Name() 和 PkgPath() 判断命名类型身份
当需要识别“是不是某个自定义类型”(比如 type UserID int 或 type Config struct{...}),就得看 Type.Name() 和 Type.PkgPath()。
立即学习“go语言免费学习笔记(深入)”;
-
t.Name()返回类型名(如果包内定义且导出则非空;未命名类型如struct{}或[]int返回空字符串) -
t.PkgPath()返回定义该类型的包路径(如"myapp/model"),可结合Name()唯一标识一个命名类型 - 安全写法:
if t.Name() != "" && t.PkgPath() != "" && t.Name() == "UserID"
处理接口与反射嵌套:间接类型展开
接口变量、指针、切片元素等可能包裹多层类型。要用 Elem()、Index() 等方法逐层解包,才能触达真实类型。
- 指针类型:先
t.Kind() == reflect.Ptr,再t.Elem()得到指向的类型 - 切片/映射/通道:用
t.Elem()获取元素类型;结构体用t.Field(i).Type查字段类型 - 接口类型:
t.Kind() == reflect.Interface表示这是一个接口类型描述;若想知其动态值类型,需从reflect.Value入手(v.Elem().Type()或v.Type()在非 nil 时才有效)
基本上就这些。反射类型判断不复杂,但容易忽略 Kind 与 Name 的分工、接口值的双层语义,以及 nil 值的安全边界。理清这三层(Kind 分类 → Name/PkgPath 识别 → Elem 展开),就能稳稳覆盖大多数场景。