反射创建结构体实例需先用reflect.TypeOf获取类型,再用reflect.New创建指针,Elem()后通过FieldByName设值;字段须导出,非零值需手动赋值,支持从map动态填充,但性能低且无编译检查。
Go 的 reflect 包允许在运行时检查类型、字段和方法,并动态创建值。要通过反射创建一个结构体实例,关键步骤是:先获取类型的 reflect.Type,再用 reflect.New 或 reflect.Zero 获取指针或零值。
reflect.New:它返回 reflect.Value,对应新分配的指针;解引用后才能得到可设置的结构体值.Elem().FieldByName("Name").Set(...) 逐个赋值示例:
type User struct {
Name string
Age int
}
t := reflect.TypeOf(User{})
v := reflect.New(t) // 返回 *User 的 reflect.Value
u := v.Elem() // 获取 User 值(可寻址、可设置)
u.FieldByName("Name").SetString("Alice")
u.FieldByName("Age").SetInt(30)
user := u.Interface().(User) // 转回原始类型
仅靠 reflect.New 得到的是零值结构体。若需
按字段名或索引批量设值,可结合 reflect.StructTag 或 map 映射实现“动态填充”。
v.NumField() 遍历所有字段,配合 v.Field(i) 和 t.Field(i) 获取字段信息与标签int 不能直接 .Set() 到 int64 字段,需用 reflect.ValueOf(int64(v)).Convert(field.Type)
小技巧:定义一个通用函数 FromMap(v interface{}, data map[string]interface{}) error,内部用反射完成字段映射,适合配置加载、API 参数绑定等场景。
除结构体外,反射也常用于动态生成容器类型:
reflect.MakeSlice:传入元素类型、长度、容量,返回 reflect.Value 表示切片reflect.MakeMap:需先用 reflect.MapOf(keyType, elemType) 构造类型,再调用 MakeMap
reflect.New 可用于任意类型(包括基本类型),如 reflect.New(reflect.TypeOf(0).Elem()) 创建 *int
注意:MakeSlice 和 MakeMap 返回的是可直接使用的值(非指针),而 New 总是返回指针的 reflect.Value。
反射虽灵活,但有明显代价:
unsafe 也不推荐绕过,破坏封装性且不稳定替代方案优先考虑代码生成(go:generate + stringer 或自定义模板)、泛型(Go 1.18+)或接口抽象,仅在真正需要“未知类型”处理时启用反射。