Go 的 reflect 包用于运行时类型和值反射,核心是区分 reflect.Type(描述类型)与 reflect.Value(封装值及操作性);需用 Kind() 判断基础类别,Name() 识别命名类型;操作前须校验 IsValid()、CanAddr()、CanSet() 等安全性条件。

在 Go 中,reflect 包是运行时获取类型和值信息的核心工具,适用于泛型尚不适用或需深度元编程的场景(如序列化、ORM、调试工具)。关键在于区分 reflect.Type 和 reflect.Value 的用途:前者描述“是什么类型”,后者代表“值本身及其可操作性”。
用 reflect.TypeOf 获取类型信息
reflect.TypeOf 接收任意接口值,返回 reflect.Type 对象,可用于判断底层类型、结构体字段、方法集等。
- 基础类型判断:用
Type.Kind()获取基本类别(如int、string、struct、ptr、slice),不是Type.Name()—— 后者对内建类型返回空字符串 - 区分命名类型与底层类型:用
Type.Name()看是否为自定义命名类型(如"Person"),用Type.Kind()看其本质(如struct) - 检查指针/切片/映射等复合类型:先用
Kind()判断大类,再用Type.Elem()或Type.Key()/Type.Elem()获取元素或键值类型
用 reflect.ValueOf 获取值信息并安全操作
reflect.ValueOf 返回 reflect.Value,它封装了值、类型及可寻址性。注意:非导出字段无法通过反射修改,且未导出字段的 Value 默认不可寻址。
- 判空用
Value.IsValid(),避免 panic;判断零值用Value.IsZero() - 取值前务必检查可寻址性:
Value.CanAddr()和Value.CanInterface()决定能否转回原始类型或取地址 - 修改值需满足:可寻址 + 可设置(
CanSet()),通常要传指针,例如reflect.ValueOf(&x).Elem()
常用类型识别与转换模式
实际中常需结合 Type 和 Value 做精准识别和转换:
立即学习“go语言免费学习笔记(深入)”;
- 判断是否为某个具体类型(如
*os.File):先Value.Type()得到Type,再用Type == reflect.TypeOf((*os.File)(nil)).Elem() - 识别结构体并遍历字段:确认
Kind() == reflect.Struct后,用NumField()和Field(i)访问字段reflect.Value,用Type.Field(i)获取标签和名称 - 安全转回原类型:用
Value.Interface()(仅当IsValid()且可转换),或配合类型断言使用,如v.Interface().(string)(需确保类型匹配)
避免常见陷阱
反射易引发 panic 或行为不符合预期,需格外注意:
- 对 nil 接口调用
reflect.ValueOf返回无效Value,必须先IsValid() -
reflect.Value的方法多数不修改原值,而是返回新Value(如Elem()、Index()) - 反射性能较低,不应在热路径频繁使用;优先考虑接口、类型断言或泛型
- 无法反射访问未导出字段的值(除非用
unsafe,不推荐)










