首页 >后端开发 >Golang >为什么 `reflect.MakeSlice` 返回一个不可寻址的 `Value`?

为什么 `reflect.MakeSlice` 返回一个不可寻址的 `Value`?

Barbara Streisand
Barbara Streisand原创
2024-11-08 05:36:011115浏览

Why does `reflect.MakeSlice` return an un-addressable `Value`?

为什么 golang Reflect.MakeSlice 返回一个不可寻址的值?

这个问题源于需要创建一个切片地址,以便在接受 a 的函数中使用指向切片的指针。

为什么 Reflect.MakeSlice 返回一个不可寻址的值?

Go 中可寻址性的松散定义是,你可以获取某个东西的地址,并保证这个地址指向有意义的地方。如果您在函数体内的堆栈上分配某些内容,则分配值的地址在某个时间点将不再可访问。因此,这个值是不可寻址的。

在大多数情况下,如果局部堆栈变量被返回或以其他方式提升到外部,Go 会将它们移动到堆中,但在运行时这不会完成。因此,CanAddr() 仅在以下情况下返回 true:

  A value is addressable if it is an element of a slice, an element of an addressable array, a field of an addressable struct, or the result of dereferencing a pointer.

所声明的类型都有一个共同点:它们保证它们所保存的内容可以从任何地方访问,并指向内存中有意义的值。自从您使用 Reflect.MakeSlice 创建了本地切片以来,您既没有切片元素,也没有指针,也没有任何其他提到的东西。不过,该切片的元素是可寻址的(因为切片的内存驻留在堆上)。

如何使用反射获取指向切片的指针

最简单的解决方案可能是使用反射.New() 创建指针(播放中的完整示例):

my := &My{}

// Create a slice to begin with
myType := reflect.TypeOf(my)
slice := reflect.MakeSlice(reflect.SliceOf(myType), 10, 10)

// Create a pointer to a slice value and set it to the slice
x := reflect.New(slice.Type())
x.Elem().Set(slice)

collection.Find(bson.M{}).All(x.Interface())

请注意其他答案也指出的 x.Interface() 。这可以防止将 x 的实际值传递给 All(),而不是 Reflect.Value。

以上是为什么 `reflect.MakeSlice` 返回一个不可寻址的 `Value`?的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn