reflect.typeof()返回变量的静态类型信息,如*int;Name()仅对导出类型返回名称,匿名结构体返回空字符串。

在 go 中,reflect 包用于运行时获取变量的类型和值信息,是实现泛型逻辑、序列化、ORM 映射等场景的基础。关键在于理解 reflect.Type 和 reflect.Value 的区别与协作方式。
获取类型信息:reflect.TypeOf()
reflect.TypeOf() 接收任意接口值,返回 reflect.Type 类型对象,描述其静态类型(编译时类型)。
- 传入指针会得到指针类型,如
reflect.TypeOf(&x)返回*int的 Type - 基础类型、结构体、切片、映射、函数等都能正确识别
- 注意:不能对 nil 指针或未初始化接口调用,否则 panic
常用类型元数据方法
reflect.Type 提供一系列方法读取类型结构:
-
Name():返回类型名(仅导出类型有名字,匿名结构体返回空字符串) :返回底层种类(<code>reflect.Struct、 reflect.Slice、reflect.Ptr等),比Name()更可靠-
String():返回完整类型字符串,如"[]string"或"main.User" -
Elem():对指针、切片、映射、通道等获取元素类型(如[]int的Elem()是int) -
Field(i int)和NumField():仅对 struct 类型有效,用于遍历字段
安全获取结构体字段信息
反射访问结构体字段前,必须确认类型是 struct 且字段可导出(首字母大写):
立即学习“go语言免费学习笔记(深入)”;
- 先用
t.Kind() == reflect.Struct判断 - 用
t.NumField()获取字段数,再循环调用t.Field(i) -
StructField中Name是字段名,Type是字段类型,Tag是结构体标签(如`json:"name"`) - 不可导出字段(小写开头)在反射中会被忽略,无法读取或设置
类型与值要分开处理
reflect.TypeOf() 只给类型,reflect.ValueOf() 才提供值操作能力。两者常配合使用:
- 想判断类型并做分支逻辑 → 用
Type.Kind() - 想读写字段、调用方法、修改内容 → 必须用
Value,且注意可寻址性(CanAddr()、CanSet()) - 从
Value可通过.Type()方法反查类型,无需重复调用reflect.TypeOf()
基本上就这些。reflect 不复杂但容易忽略类型与值的边界,多数误用都源于混淆 TypeOf 和 ValueOf 的职责。