
本文介绍如何使用 go 的 reflect 包在运行时获取任意结构体(或接口)类型的名称,重点解决通过指针、值或接口传递时准确提取类型名的问题,并对比 name() 与 string() 方法的适用场景。
在 Go 中,类型信息在编译期确定,但若需在运行时动态获取结构体或接口的名称(例如用于日志、序列化、泛型模拟或 ORM 映射),必须借助 reflect 包。核心思路是:将任意 interface{} 值转换为 reflect.Value,再通过其类型(Type())提取名称。
以下是一个完整、健壮的实现示例:
package main
import (
"fmt"
"reflect"
"./crud"
)
func get_struct_name(value interface{}) {
v := reflect.ValueOf(value)
// 处理 nil 接口或未初始化值
if !v.IsValid() {
fmt.Println("nil")
return
}
t := v.Type()
// 若传入的是指针(如 &crud.User{}),需解引用以获取实际结构体类型名
for t.Kind() == reflect.Ptr || t.Kind() == reflect.Slice || t.Kind() == reflect.Map {
t = t.Elem()
}
// 推荐使用 t.Name() 获取未限定的类型名(如 "User")
// 若需完整路径(如 "crud.User"),则用 t.String()
name := t.Name()
if name == "" {
// 非命名类型(如 struct{}、[]int、map[string]int)无 Name(),String() 更合适
name = t.String()
}
fmt.Println(name)
}
func main() {
get_struct_name(&crud.User{}) // 输出: User
get_struct_name(crud.User{}) // 输出: User
get_struct_name([]int{}) // 输出: []int(因 []int 无 Name,String() 自动生效)
}关键要点说明:
- ✅ reflect.ValueOf(value).Type().Name() 仅对命名类型(如 type User struct {...})返回非空字符串;匿名结构体、切片、映射等返回空字符串。
- ✅ 使用 t.Elem() 递归解引用可统一处理 *T、[]T、map[K]V 等复合类型,确保获取到最内层的底层类型名。
- ✅ t.String() 返回完整、可读的类型描述(含包路径和结构体字面量),适用于调试和通用类型识别;而 t.Name() 更适合获取简洁、语义化的类型标识(如 ORM 表名映射)。
- ⚠️ 注意:reflect 带来一定性能开销,不建议在高频热路径中频繁调用;生产环境应结合缓存(如 sync.Map 缓存 reflect.Type → 名称映射)优化。
综上,get_struct_name 函数兼顾了安全性、通用性与可读性,是 Go 运行时类型元信息获取的标准实践。










