
在Go语言中,使用反射创建新实例时,reflect.New 是一个常用方法。它会为指定的类型分配内存,并返回一个指向新分配零值的指针。理解其内存分配行为对性能和正确性都很重要。
reflect.New 基本用法
reflect.New 接受一个 reflect.Type 或类型对应的 reflect.Value.Type(),返回 *T 类型的 reflect.Value,其中 T 是原始类型。
例如:
func createInstance(t reflect.Type) reflect.Value {
return reflect.New(t)
}
假设你有一个结构体:
立即学习“go语言免费学习笔记(深入)”;
type User struct {
Name string
Age int
}
typ := reflect.TypeOf(User{})
v := reflect.New(typ)
// v 是 reflect.Value,表示 *User 类型的指针,指向新分配的零值 User{}
内存分配机制
reflect.New 实际上等价于在Go中使用 &T{} 的方式创建一个指向新零值的指针,但它在运行时动态完成。
它会调用运行时的 new(T) 操作,分配足够的内存来存储类型 T 的零值 返回的是指向该内存的指针,封装为 reflect.Value 分配的内存位于堆上,由Go的垃圾回收器管理
例如:
v := reflect.New(reflect.TypeOf(User{}))
// 等价于:
// &User{}
// 或者:
// new(User)
获取可设置的实例
通过 reflect.New 得到的 reflect.Value 是一个指针。要修改其指向的值,需要使用 Elem() 获取指针指向的值对象:
v := reflect.New(reflect.TypeOf(User{}))
e := v.Elem() // 获取指向的结构体值,可设置
e.Field(0).SetString(“Alice”)
e.Field(1).SetInt(30)
// 转回接口或具体类型
user := v.Interface().(*User)
fmt.Printf(“%+vn”, user) // &{Name:Alice Age:30}
注意:必须调用 Elem() 才能获得可设置(addressable)的字段操作权限。
与 reflect.Zero 的区别
reflect.New 分配内存,返回指向新零值的指针 reflect.Zero 不分配内存,直接返回对应类型的零值(不可寻址)
如果你需要一个可修改的实例,应使用 reflect.New。reflect.Zero 适合用于默认值传递或类型匹配场景。
基本上就这些。reflect.New 的内存分配行为是安全且直观的,相当于动态执行 new(T),适合在需要动态创建实例的场景中使用,比如ORM、配置解析、通用工厂等。虽然反射有性能开销,但在必要时是有效的工具。
以上就是Golang反射创建新实例 reflect.New内存分配的详细内容,更多请关注创想鸟其它相关文章!
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1404781.html
微信扫一扫
支付宝扫一扫