一尘不染

将字符串切片转换为指向字符串的指针切片

go

我想将字符串切片转换为指向字符串的指针切片

values1 := []string{"a", "b", "c"}
var values2 []*string
for _, v := range values1 {
    fmt.Printf("%p | %T\n", v, v)
    values2 = append(values2, &v)
}
fmt.Println(values2)

%!p(string = a)=>字符串
%!p(string = b)=>字符串
%!p(string = c)=>字符串
[0xc42000e1d0 0xc42000e1d0 0xc42000e1d0]

据我了解,
我的变量v似乎是一个字符串,而不是指向字符串的指针。
因此v应从values1迭代时复制。

显然我不正确,因为v地址仍然相同0xc42000e1d0。如果值不是指针,该
如何v更改?

快速解决方案使用:

values1 := []string{"a", "b", "c"}
var values2 []*string
for i, _ := range values1 {
    values2 = append(values2, &values1[i])
}

阅读 214

收藏
2020-07-02

共1个答案

一尘不染

第一个版本不起作用,因为只有一个循环变量v每次迭代 都有 相同的地址
。在每次迭代中都会分配循环变量,该变量会更改其值,但不会更改其地址,并且您会在每次迭代中附加相同的地址。

可能的解决方案是您提出的解决方案:在原始切片上附加适当切片元素的地址:

for i := range values1 {
    values2 = append(values2, &values1[i])
}

此方法有效,但请注意,values2其中包含指向的后备数组的地址values1,因此的后备数组values1将保留在内存中(不会收集垃圾),直到values1values2变量均不可访问。另请注意,由于的元素指向的元素,因此修改的元素values1values2间接影响。values2``values1

另一种解决方案是创建临时局部变量,并附加这些变量的地址:

for _, v := range values1 {
    v2 := v
    values2 = append(values2, &v2)
}

这样,您的分values2片将无法values1避免收集垃圾,也values1不会修改中的值values2,它们将是独立的。

2020-07-02