# Go语言中for range的"坑"

3 min read
文章目录

前言

Go 中的for range组合可以和方便的实现对一个数组或切片进行遍历,但是在某些情况下使用for range时很可能就会被"坑",下面用一段代码来模拟下:

func main() {
arr1 := []int{1, 2, 3}
arr2 := make([]*int, len(arr1))
for i, v := range arr1 {
arr2[i] = &v
}
for _, v := range arr2 {
fmt.Println(*v)
}
}

代码解析:

  1. 创建一个int slice,变量名为arr1并初始化 1,2,3 作为切片的值。
  2. 创建一个*int slice,变量名为arr2
  3. 通过for range遍历arr1,然后获取每一个元素的指针,赋值到对应arr2中。
  4. 逐行打印arr2中每个元素的值。

从代码上看,打印出来的结果应该是

1
2
3

然而真正的结果是

3
3
3

原因

因为for range在遍历值类型时,其中的v变量是一个的拷贝,当使用&获取指针时,实际上是获取到v这个临时变量的指针,而v变量在for range中只会创建一次,之后循环中会被一直重复使用,所以在arr2赋值的时候其实都是v变量的指针,而&v最终会指向arr1最后一个元素的值拷贝。

来看看下面这个代码,用for i来模拟for range,这样更易于理解:

func main() {
arr1 := []int{1, 2, 3}
arr2 := make([]*int, len(arr1))
var v int
for i:=0;i<len(arr1);i++ {
v = arr1[i]
arr2[i] = &v
}
for _, v := range arr2 {
fmt.Println(*v)
}
}

解决方案

  1. 传递原始指针
func main() {
arr1 := []int{1, 2, 3}
arr2 := make([]*int, len(arr1))
for i := range arr1 {
arr2[i] = &arr1[i]
}
for _, v := range arr2 {
fmt.Println(*v)
}
}
  1. 使用临时变量
func main() {
arr1 := []int{1, 2, 3}
arr2 := make([]*int, len(arr1))
for i, v := range arr1 {
t := v
arr2[i] = &t
}
for _, v := range arr2 {
fmt.Println(*v)
}
}
  1. 使用闭包
func main() {
arr1 := []int{1, 2, 3}
arr2 := make([]*int, len(arr1))
for i, v := range arr1 {
func(v int){
arr2[i] = &v
}(v)
}
for _, v := range arr2 {
fmt.Println(*v)
}
}

官方提示

由于这一问题过于普遍,Golang甚至将其写入了文档的『常见错误』部分:文档


More Posts

# java8下HTTP代理身份验证设置

3 min read

由于公司内部应用要调用钉钉的 API,但是钉钉 API 有一个 IP 白名单限制,而公司的外网 IP 经常变动,每次变动都需要在钉钉的后台配置一个 IP,在开发环境调试非常的麻烦,于是就让运维在一台外网服务器上搭建了一个HTTP代理服务,通过代理服务器转发,只需要设置代理服务器的外网 IP 就可以避免之前的问题了。

阅读

评论区