在 go 语言中,反射可通过 reflect.type 和 reflect.value 获取结构体字段信息、设置字段值及调用方法。1. 获取字段信息:使用 typeof().numfield() 获取字段数量,typeof().field(i) 获取字段名和标签(tag);2. 设置字段值:需传入结构体指针并使用 elem() 获取可写 value,通过 fieldbyname() 定位字段并调用 setstring() 等方法修改值;3. 调用方法:使用 methodbyname() 获取方法并以 call() 执行,参数以 []reflect.value 或 nil 传递;常见用途包括 orm 框架、json 序列化/反序列化、配置解析、校验器及插件系统等,反射虽强大但应谨慎使用以避免性能问题。
在 Go 语言中,反射(reflect)是一个非常强大的工具,尤其在处理结构体时,它可以帮助我们动态地获取结构体字段、方法,甚至修改其值。这种能力在实现通用库、ORM 框架、序列化/反序列化等场景中尤为重要。
如何用反射获取结构体字段信息?
当我们拿到一个结构体的 reflect.Type 时,可以通过 .NumField() 和 .Field(i) 方法来遍历所有字段。
举个例子:
立即学习“go语言免费学习笔记(深入)”;
type User struct { Name string Age int `json:"user_age"` }
使用反射时,可以这样获取字段名和标签:
t := reflect.TypeOf(User{}) for i := 0; i < t.NumField(); i++ { field := t.Field(i) fmt.Println("字段名:", field.Name) fmt.Println("标签值:", field.Tag.Get("json")) }
- 字段名必须是导出的(首字母大写),否则反射拿不到。
- 标签(tag)常用于 JSON、GORM 等框架做映射。
反射如何设置结构体字段的值?
如果想通过反射修改结构体字段的值,需要注意以下几点:
- 必须传入结构体的指针
- 要使用 reflect.ValueOf() 获取可写的 Value
- 字段也必须是导出的
示例代码如下:
u := &User{} v := reflect.ValueOf(u).Elem() nameField := v.Type().Field(0) if nameField.Name == "Name" { fieldValue := v.FieldByName("Name") if fieldValue.CanSet() { fieldValue.SetString("Tom") } }
注意:如果你传的是值类型而不是指针,Elem() 会报错;如果字段不可写(比如未导出或不是地址),CanSet() 会返回 false。
反射调用结构体的方法
反射不仅能读写字段,还能调用方法。前提是方法是导出的,并且参数匹配。
例如:
func (u User) SayHello() { fmt.Println("Hello", u.Name) }
通过反射调用:
u := &User{Name: "Jerry"} v := reflect.ValueOf(u) method := v.MethodByName("SayHello") if method.IsValid() { method.Call(nil) // 无参数就传 nil }
- MethodByName() 返回的是 reflect.Value 类型
- 参数要以 []reflect.Value{} 的形式传入,即使没有参数也要用 nil
实际开发中的常见用途
反射在结构体上的应用很广,下面是一些常见的实际用途:
- ORM 框架:自动将数据库字段映射到结构体字段,依赖 tag 和字段名。
- JSON 序列化/反序列化:标准库 encoding/json 内部大量使用反射解析结构体。
- 配置解析:从 YAML、TOML 文件加载配置到结构体中。
- 校验器:比如 validator 库通过 tag 解析校验规则。
- 插件系统:通过反射查找并调用插件接口的方法。
这些功能之所以能实现,靠的就是反射对结构体字段和方法的动态访问能力。
基本上就这些了。Go 的反射机制虽然强大,但用起来也得小心,尤其是在性能敏感的地方尽量避免频繁使用。不过对于结构体的处理来说,它确实提供了很多灵活性。