使用 reflect.valueof 获取结构体字段前必须先解引用:v := reflect.valueof(&s).elem(),否则 numfield() 会 panic;需检查指针非 nil 且仅对 struct 和非 nil ptr 递归;导出字段才可 interface(),私有字段需跳过或用其他方式处理。

用 reflect.ValueOf 获取结构体字段前必须解引用
直接对指针调用 reflect.ValueOf 得到的是指针类型,.NumField() 会 panic:「panic: reflect: NumField of non-struct type」。必须先 .Elem() 解引用,否则连字段数都读不到。
- 正确写法:
v := reflect.ValueOf(&s).Elem()(s是结构体变量) - 错误写法:
v := reflect.ValueOf(s)或v := reflect.ValueOf(&s)(后者是*T类型,不是T) - 如果输入可能是 nil 指针,
.Elem()前要加.Kind() == reflect.Ptr && !v.IsNil()判断,否则 panic
遍历字段时区分导出与非导出字段
Go 反射只能读取导出字段(首字母大写),非导出字段的 .CanInterface() 返回 false,.Interface() 会 panic:「reflect: call of reflect.Value.Interface on unexported field」。
- 检查是否可读:
if !f.CanInterface() { continue } - 常见误操作:没加判断就直接
f.Interface(),尤其在通用序列化工具里容易炸 - 若需处理私有字段(如测试、调试),改用
unsafe或 struct tags 显式声明,反射本身不支持绕过导出限制
递归进入嵌套结构体或指针字段要控制入口条件
无限递归通常源于没拦住指针循环引用(比如 A 字段指向 B,B 字段又指向 A)或没过滤掉非结构体类型(如 map、slice、函数、channel)。
- 只对
reflect.Struct和非 nil 的reflect.Ptr类型递归 - 遇到
reflect.Ptr先判空:if v.Kind() == reflect.Ptr && !v.IsNil() { v = v.Elem() },再检查是否为 struct - 简单去重可用
map[uintptr]bool记录unsafe.Pointer(v.UnsafeAddr()),但注意:UnsafeAddr()对非地址able 值(如 map value)会 panic
reflect.Value.Interface() 的类型断言风险
从反射值取回原始 Go 值时,.Interface() 返回 interface{},直接类型断言可能 panic,尤其字段类型不确定时(比如 struct 字段是 interface{} 或泛型参数)。
- 安全做法:用
switch v.Kind()分支处理基本类型,或用if ok := v.CanInterface(); ok { ... }+v.Interface() - 常见坑:
v.Interface().(string)在字段是*string时失败,应先v.Elem().Interface().(string)(且确保是 ptr) - 性能提示:频繁
.Interface()会触发接口分配,高频场景建议用.String()、.Int()等原生方法代替










