Go中递归获取嵌套结构体字段信息需用reflect包深度优先遍历:判断Struct类型、跳过非导出字段、安全解引用指针/接口、拼接字段路径,对基础类型停止递归并记录路径与值,全程校验IsValid()等防panic。

在 Go 中处理嵌套结构体并递归获取字段信息,核心是使用 reflect 包配合深度优先遍历。关键在于区分结构体类型、跳过非导出字段、正确处理指针/接口/切片等中间类型,并安全提取字段名与值。
判断是否为结构体并进入递归
通过 reflect.TypeOf 获取类型后,用 .Kind() 判断是否为 reflect.Struct;若为指针或接口,先 Elem() 解引用(需检查是否可寻址/有效)。对每个字段调用 .Type() 和 .Interface() 获取其类型与值,再递归处理。
- 只处理导出字段(首字母大写),
field.IsExported()可直接判断 - 遇到
nil指针或未初始化接口,跳过或返回零值,避免 panic - 用
field.Anonymous识别匿名字段,可选择是否展开(如 ORM 映射常用)
提取字段路径与值的组合信息
递归时传入当前字段路径(如 "User.Profile.Address.Street"),每深入一层拼接字段名。对基础类型(int、string、bool 等)停止递归,记录路径+值;对 map/slice 则需额外循环索引,但通常只展开 struct 层级。
- 路径分隔符建议用
.,符合 Go 命名习惯,也便于后续映射到 JSON key - 对 slice,可记录
Path[0]、Path[1]等形式,或统一转为Path.#表示任意元素 - 用
fmt.Sprintf("%v", val.Interface())安全转字符串值,避免类型断言错误
规避常见 panic 场景
reflect 操作极易 panic:空接口、未导出字段、不可寻址值、未初始化指针。应在每次取值前加保护:
立即学习“go语言免费学习笔记(深入)”;
- 用
val.IsValid()判定值是否有效 - 用
val.CanInterface()确保能安全转为 interface{} - 对指针类型,先
val.Elem()再检查IsValid(),而非直接解引用 - 避免对 map/slice 直接调用
.NumField()—— 它们不是 struct,会 panic
简化封装:一个可复用的递归遍历函数
定义函数签名如 WalkStruct(v interface{}, fn func(path string, value interface{})),内部做类型分发。struct 字段递归调用自身;基础类型直接执行回调;slice/map 迭代元素并带上下标路径;其余类型(chan、func、unsafe 等)忽略。
- 不强制要求输入为指针,函数内自动
reflect.ValueOf(v).DeepCopy()或取地址处理 - 支持提前退出:回调函数返回
false即中断递归(需修改为带返回值的 fn) - 可选参数控制是否展开匿名字段、是否忽略空值、是否限制递归深度










