我是靠谱客的博主 魁梧蓝天,这篇文章主要介绍详解Go语言中for range的"坑",现在分享给大家,希望可以做个参考。

前言

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

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
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) } }

代码解析:

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

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

1
2
3

然而真正的结果是

3
3
3

原因

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

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

复制代码
1
2
3
4
5
6
7
8
9
10
11
12
13
14
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
2
3
4
5
6
7
8
9
10
11
12
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
2
3
4
5
6
7
8
9
10
11
12
13
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
2
3
4
5
6
7
8
9
10
11
12
13
14
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甚至将其写入了文档的『常见错误』部分:文档

到此这篇关于详解Go语言中for range的"坑"的文章就介绍到这了,更多相关Go语言for range内容请搜索靠谱客以前的文章或继续浏览下面的相关文章希望大家以后多多支持靠谱客!

最后

以上就是魁梧蓝天最近收集整理的关于详解Go语言中for range的"坑"的全部内容,更多相关详解Go语言中for内容请搜索靠谱客的其他文章。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(164)

评论列表共有 0 条评论

立即
投稿
返回
顶部