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) } }
代码解析:git
int slice
,变量名为arr1
并初始化 1,2,3 做为切片的值。*int slice
,变量名为arr2
。for range
遍历arr1
,而后获取每个元素的指针,赋值到对应arr2
中。arr2
中每一个元素的值。从代码上看,打印出来的结果应该是github
1 2 3
然而真正的结果是golang
3 3 3
由于for range
在遍历值类型
时,其中的v
变量是一个值
的拷贝,当使用&
获取指针时,其实是获取到v
这个临时变量的指针,而v
变量在for range
中只会建立一次,以后循环中会被一直重复使用,因此在arr2
赋值的时候其实都是v
变量的指针,而&v
最终会指向arr1
最后一个元素的值拷贝。docker
来看看下面这个代码,用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) } }
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) } }
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) } }
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甚至将其写入了文档的『常见错误』部分:文档闭包
我是 MonkeyWie,欢迎扫码👇👇关注!不按期在公众号中分享JAVA
、Golang
、前端
、docker
、k8s
等干货知识。