在 Go 中获取嵌套结构体类型信息需用 reflect 包递归解析:先用 reflect.TypeOf 获取顶层类型,对每个字段调用 Field(i).Type,遇 struct 则继续遍历,指针等需 Elem() 解包,未导出字段不可见。

在 Go 中获取嵌套结构体的类型信息,核心是借助 reflect 包逐层解析字段类型。关键不是“一次性拿到最内层”,而是理解类型层级如何展开、字段如何递归访问。
用 reflect.TypeOf 获取顶层结构体类型
先通过 reflect.TypeOf 获得接口值的 reflect.Type,注意传入指针可保留结构体本身(而非 interface{} 的底层类型):
- 若变量是结构体实例:
reflect.TypeOf(v)返回其类型;若想看到导出字段的完整结构,通常传&v更稳妥 - 对嵌套字段,需先用
.Field(i)取字段,再调用.Type获取该字段的类型 - 例如:
user.Address.Street对应的类型,需先取User的Address字段类型(可能是Address结构体),再从中取Street字段类型(如string)
递归遍历结构体字段以分析完整嵌套层次
Go 的结构体嵌套没有深度限制,需手动递归处理 reflect.StructField。每次遇到 Kind() == reflect.Struct 就继续深入:
- 检查字段类型
t.Kind()是否为reflect.Struct - 若是,用
t.NumField()遍历其所有字段,对每个字段重复该过程 - 可用栈或递归函数记录路径,例如
["User", "Address", "City"]表示嵌套路径 - 注意:未导出字段(小写开头)在反射中不可见,
NumField()不会返回它们
区分嵌套结构体与指针/切片等包装类型
实际类型常带间接性——比如 *Address、[]Order、map[string]Product。需先“解包”再判断:
立即学习“go语言免费学习笔记(深入)”;
- 用
t.Elem()处理指针、切片、map、channel 等有元素类型的类型(Elem()返回其基类型) - 用
t.Key()和t.Elem()分别获取 map 的 key/value 类型 - 嵌套中的
*Address需先Elem()得到Address类型,再判断是否为 struct 并继续展开 - 避免直接对非 struct 类型调用
NumField(),会 panic
打印或序列化嵌套类型结构的实用技巧
调试时快速看清类型树,可用递归格式化输出;生产中可转为 JSON Schema 或用于生成文档:
- 定义一个递归函数,接收
reflect.Type和当前缩进/路径,输出字段名、类型名、是否为 struct - 对匿名字段(嵌入字段),
Field(i).Anonymous为 true,可标记为 “embedded” - 用
t.Name()和t.PkgPath()判断是否为命名类型,区分type UserID int和原始int - 注意:
reflect无法还原泛型实参(如Map[string]int在运行时擦除为Map),Go 1.18+ 的类型参数需结合 AST 分析










