反射创建结构体实例需先用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)获取字段信息与标签 - 支持从 map[string]interface{} 构建:遍历 map 键,匹配结构体字段名(忽略大小写或按 tag 指定 key)
- 注意类型转换:map 中的
int不能直接.Set()到int64字段,需用reflect.ValueOf(int64(v)).Convert(field.Type)
小技巧:定义一个通用函数 FromMap(v interface{}, data map[string]interface{}) error,内部用反射完成字段映射,适合配置加载、API 参数绑定等场景。
立即学习“go语言免费学习笔记(深入)”;
反射创建切片、映射和指针
除结构体外,反射也常用于动态生成容器类型:
-
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。
安全与性能提醒
反射虽灵活,但有明显代价:
- 无编译期类型检查:字段名写错、类型不匹配会在运行时报 panic,建议搭配单元测试或封装校验逻辑
- 性能开销大:比直接调用慢 10–100 倍,高频路径(如 HTTP handler 内部)应避免反复反射
-
无法操作未导出字段:即使使用
unsafe也不推荐绕过,破坏封装性且不稳定
替代方案优先考虑代码生成(go:generate + stringer 或自定义模板)、泛型(Go 1.18+)或接口抽象,仅在真正需要“未知类型”处理时启用反射。










