首页 >后端开发 >Golang >为什么从 Go Slice 复制会导致内存地址重用,如何修复?

为什么从 Go Slice 复制会导致内存地址重用,如何修复?

Barbara Streisand
Barbara Streisand原创
2024-12-25 21:34:11390浏览

Why Does Copying from a Go Slice Result in Memory Address Reuse, and How Can It Be Fixed?

从 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中文网其他相关文章!

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