当前位置: 首页 > 编程笔记 >

详解Go语言中for range的"坑"

鲍向笛
2023-03-14
本文向大家介绍详解Go语言中for range的"坑",包括了详解Go语言中for range的"坑"的使用技巧和注意事项,需要的朋友参考一下

前言

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)
  }
}

代码解析:

  • 创建一个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,这样更易于理解:

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甚至将其写入了文档的『常见错误』部分:文档

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

 类似资料:
  • 本文向大家介绍Go语言method详解,包括了Go语言method详解的使用技巧和注意事项,需要的朋友参考一下 前面两章我们介绍了函数和struct,那你是否想过函数当作struct的字段一样来处理呢?今天我们就讲解一下函数的另一种形态,带有接收者的函数,我们称为method method 现在假设有这么一个场景,你定义了一个struct叫做长方形,你现在想要计算他的面积,那么按照我们一般的思路应

  • 本文向大家介绍Go语言interface详解,包括了Go语言interface详解的使用技巧和注意事项,需要的朋友参考一下 interface Go语言里面设计最精妙的应该算interface,它让面向对象,内容组织实现非常的方便,当你看完这一章,你就会被interface的巧妙设计所折服。 什么是interface 简单的说,interface是一组method的组合,我们通过interface

  • 本文向大家介绍详解go语言的并发,包括了详解go语言的并发的使用技巧和注意事项,需要的朋友参考一下 1、启动go语言的协程 2、runtime.Goexit()方法。立即终止当前的协程 3、runtime.GOMAXPROCS()表示go使用几个cpu执行代码 4、管道定义和创建管道 5、管道的缓冲 6、关闭管道和接受关闭管道的信号 7、只读管道和只写管道和生产者和消费者模型 8、Timer定时器

  • 本文向大家介绍Go语言中new()和 make()的区别详解,包括了Go语言中new()和 make()的区别详解的使用技巧和注意事项,需要的朋友参考一下 概述 Go 语言中的 new 和 make 一直是新手比较容易混淆的东西,咋一看很相似。不过解释两者之间的不同也非常容易。 new 的主要特性 首先 new 是内建函数,你可以从 http://golang.org/pkg/builtin/#n

  • 本文向大家介绍Go语言中你不知道的Interface详解,包括了Go语言中你不知道的Interface详解的使用技巧和注意事项,需要的朋友参考一下 前言 最近在看Go语言的面向对象的知识点时,发现它的面向对象能力全靠 interface 撑着,而且它的 interface 还与我们以前知道的 interface 完全不同。故而整个过程不断的思考为什么要如此设计?这样设计给我们带来了什么影响? in

  • 本文向大家介绍Go语言并发技术详解,包括了Go语言并发技术详解的使用技巧和注意事项,需要的朋友参考一下 有人把Go比作21世纪的C语言,第一是因为Go语言设计简单,第二,21世纪最重要的就是并行程序设计,而Go从语言层面就支持了并行。 goroutine goroutine是Go并行设计的核心。goroutine说到底其实就是线程,但是它比线程更小,十几个goroutine可能体现在底层就是五六个