反射获取切片元素类型须调用t.elem();匿名字段需手动递归处理;未导出字段调interface()会panic;[]mytype不可convert为[]interface{},需逐个set;所有操作均需检查导出性、可设置性等安全条件。

反射获取切片元素类型时,reflect.TypeOf 返回的是指针而非实际元素类型
常见错误是直接对切片变量调用 reflect.TypeOf,结果拿到的是 *[]MyStruct 或 []MyStruct 类型本身,而不是 MyStruct。必须先取 Elem() 才能向下穿透。
- 对切片类型:先
t := reflect.TypeOf(slice),再t.Elem()得到元素类型(注意不是t.Elem().Elem()) - 若切片是 interface{} 传入的,需先
reflect.ValueOf(v).Kind() == reflect.Slice判断,再用.Type().Elem() - 自定义类型嵌套时(如
[]*MyStruct),t.Elem()返回*MyStruct,还需再调一次.Elem()才到MyStruct
递归解析结构体字段时,reflect.StructField.Anonymous 影响字段可见性
匿名字段(内嵌结构体)默认会被展开,但反射遍历时不会自动“扁平化”——你得手动判断 Anonymous 并递归调用,否则会漏掉深层字段。
- 遍历
reflect.Value.Field(i)前,先检查typ.Field(i).Anonymous - 若为 true,且该字段类型是 struct,则应进入递归;否则按普通字段处理
- 注意:
jsontag 不影响反射行为,但如果你在做序列化映射,需额外提取structField.Tag.Get("json") - 递归入口要限制深度,避免循环引用导致栈溢出(比如两个 struct 互相嵌套指针)
reflect.Value.Interface() 在非导出字段上 panic: "cannot interface with unexported field"
这是最常踩的坑:哪怕字段是匿名嵌入的,只要名字小写(未导出),reflect.Value.Interface() 就会直接 panic,而不是返回零值或跳过。
- 安全做法是:只对
field.CanInterface()为 true 的字段调用.Interface() - 想读未导出字段?不行——Go 反射不提供绕过导出规则的接口,这是语言设计约束,不是技巧问题
- 替代方案:改用
field.Kind()+field.Type().Name()做类型识别,用field.String()或field.Int()等方法取基本值(仅限基础类型) - 如果目标是深拷贝或序列化,建议提前约定所有需反射的字段必须导出,否则反射层无法兜底
处理 []interface{} 和 []MyType 混用时,reflect.Value.Convert() 会报 "cannot convert" 错误
Go 中切片类型不兼容,[]string 不能直接转成 []interface{},反射里也不能靠 .Convert() 强转——类型系统不允许。
立即学习“go语言免费学习笔记(深入)”;
- 正确做法:用
reflect.MakeSlice创建目标类型切片,再逐个.Index(i).Set(src.Index(i)) - 尤其注意:如果源切片是
[]*T,目标要[]interface{},则每个src.Index(i).Interface()必须可导出,否则又掉进上一个坑 - 性能敏感场景避免反射转换,优先用类型断言或生成专用转换函数(如通过
go:generate) - 第三方库如
github.com/mitchellh/mapstructure内部也遵循这套规则,只是封装了容错逻辑
递归+反射+自定义类型组合起来,真正麻烦的不是语法,而是每层都要检查导出性、可设置性、是否为指针、是否为空接口——这些判断漏掉任意一个,运行时就崩。










