我想将字符串切片转换为指向字符串的指针切片
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
values1
显然我不正确,因为v地址仍然相同0xc42000e1d0。如果值不是指针,该 如何v更改?
0xc42000e1d0
快速解决方案使用:
values1 := []string{"a", "b", "c"} var values2 []*string for i, _ := range values1 { values2 = append(values2, &values1[i]) }
第一个版本不起作用,因为只有一个循环变量v, 每次迭代 都有 相同的地址 。在每次迭代中都会分配循环变量,该变量会更改其值,但不会更改其地址,并且您会在每次迭代中附加相同的地址。
可能的解决方案是您提出的解决方案:在原始切片上附加适当切片元素的地址:
for i := range values1 { values2 = append(values2, &values1[i]) }
此方法有效,但请注意,values2其中包含指向的后备数组的地址values1,因此的后备数组values1将保留在内存中(不会收集垃圾),直到values1和values2变量均不可访问。另请注意,由于的元素指向的元素,因此修改的元素values1将values2间接影响。values2``values1
values2
values2``values1
另一种解决方案是创建临时局部变量,并附加这些变量的地址:
for _, v := range values1 { v2 := v values2 = append(values2, &v2) }
这样,您的分values2片将无法values1避免收集垃圾,也values1不会修改中的值values2,它们将是独立的。