从 Golang 切片复制时的内存地址重用
Go 编程中的一个常见误解是从切片复制内存地址。考虑以下情况:
您有一个 Model 接口和一个实现它的 Region 结构。该接口是在 Region 结构体的指针上实现的。您还有一个 Regions 集合,它是 Region 对象的切片。您有一个将 Regions 对象转换为 []Model 的方法:
// Regions is the collection of the Region model type Regions []Region // Returns the model collection as a list of models func (coll *Regions) ToModelList() []Model { output := make([]Model, len(*coll)) for idx, item := range *coll { output[idx] = &item } return output }
当您运行此代码时,您会期望在输出切片中获得每个模型的不同地址。但是,您最终会多次输出指向 Region 的第一个指针。
要解决此问题,请考虑原始方法中的循环变量项。对于循环的每次迭代都是相同的,因此生成的切片包含对相同基础数据的引用。
另一方面,工作解决方案使用 i := (*coll)[idx] 来为每次迭代创建一个新变量。这确保了输出切片中的每个模型都有唯一的内存地址。
为了更好地理解,请考虑以下代码片段:
func main() { coll := []int{5, 10, 15} for i, v := range coll { fmt.Printf("This one is always the same; %v\n", &v) fmt.Println("This one is 4 bytes larger each iteration; %v\n", &coll[i]) } }
此代码演示了 v 的内存地址在整个循环中保持不变,而 coll[i] 的内存地址随着每次迭代而增加。这种行为是由于 v 是循环变量,它保持不变,而 coll[i] 是每次迭代的新变量。
以上是为什么从 Go Slice 复制会导致内存地址重用,如何修复?的详细内容。更多信息请关注PHP中文网其他相关文章!