[英]Get pointers for values in the slice
有没有办法获得指向切片中实际值的指针? 我在地图上有一棵报价树:
map[uint64]map[uint16][]offer
为了通过 id(从 1 到 n)快速访问,我需要一片指针:
[]*offer
实际报价位于第一棵树中,切片中应该是指向实际报价的指针。
我无法遍历树并收集实际值的指针,因为这个切片的range
给了我一个副本,但我总共有超过数十亿个这样的结构,副本会导致我的内存浪费:一个结构导致 40 字节到 100+ GB 以及未来更多。 我还需要将树中的值片段存储为连续的,以便实际查找要约。
可能有什么方法可以使用unsafe
或reflect
包来获取这些指针?
指针索引仅构建一次,并且两个结构都是不可变的,仅用于查找报价。
更新:我完全错了。 获取切片中元素的指针没有问题。 我的代码是:
var offers []offer // actually there was about 50gb of offers
for i := range offers {
currentOffer := offers[i]
s.Relations[currentOffer.Id] = ¤tOffer
}
在此片段之后,应用程序消耗的 ram 总量变为 100+gb。 我立即认为从值的切片中获取元素会给我一个该值的副本,但这是错误的。 我只是自己将原始值复制到currentOffer
变量中。 一个小错误带走了 50GB 的内存。
实际上,此代码段按预期工作正常:
for i := range offers {
s.Relations[offers[i].Id] = &offers[i]
}
经过一番调查,我得出了以下代码:
package main
import (
"fmt"
"reflect"
"unsafe"
)
type offer struct {
id uint64
}
func main() {
sl := []offer{{1}, {id: 2}}
size := unsafe.Sizeof(offer{})
header := (*reflect.SliceHeader)(unsafe.Pointer(&sl))
for i := 0; i < len(sl); i++ {
offset := uintptr(i) * size
ptr := header.Data + offset
o := (*offer)(unsafe.Pointer(ptr))
o.id = 5
fmt.Println(o)
}
for _, o := range sl {
fmt.Println(o.id)
}
}
游乐场: https : //play.golang.org/p/OM3i84cKAB_7
输出:
&{5}
&{5}
5
5
这段代码说明了对切片中实际数据的访问,而不是复制的数据。
更新:通过不安全指针获取指针与按索引从切片中获取常规指针具有相同的效果。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.