Go 语言 for-range 的两个坑,你踩过吗?

简介: 先看看下面的例子,你知道最终输出的结果是什么吗?不知道的同学,大家可以在 https://play.golang.org/ 这里尝试运行一下。

坑一:迭代时协程引用索引和值

先看看下面的例子,你知道最终输出的结果是什么吗?

package main

import (
    "fmt"
    "time"
)

func main() {
   
    var m = []int{
   1, 3, 5}
    for i, v := range m {
   
        go func() {
   
            fmt.Println(i, v)
        }()
    }
    time.Sleep(time.Second)
}

不知道的同学,大家可以在 https://play.golang.org/ 这里尝试运行一下。

正确答案是:

2 5
2 5
2 5

这是为啥?三个 goroutine 都是输出了最后迭代的索引和值。

我觉得,理解清楚以下两点就可以了:

  1. 闭包内引用了外部变量 i 和 v,三个协程都引用了
  2. 协程运行时,循环可能已经结束了

要想解决这个问题,可以改成闭包传参的形式:

func main() {
   
    var m = []int{
   1, 3, 5}
    for i, v := range m {
   
        go func(_i, _v int) {
   
            fmt.Println(_i, _v)
        }(i, v)
    }
    time.Sleep(time.Second)
}

也可以让每一轮循环都用新的变量:

func main() {
   
    var m = []int{
   1, 3, 5}
    for i, v := range m {
   
        _i := i
        _v := v
        go func() {
   
            fmt.Println(_i, _v)
        }()
    }
    time.Sleep(time.Second)
}

运行结果就符合预期了:

0 1
1 3
2 5

坑二:迭代时值为原先迭代对象的拷贝

package main

import (
    "fmt"
)

func main() {
   
    var m = []int{
   1, 3, 5}
    for i, v := range m {
   
        if i == 1 {
   
            v = 2
        }
        fmt.Println(i, v)
    }
    fmt.Println(m)
}

这个输出是啥?

0 1
1 2 // 明明改成 2 了,
2 5
[1 3 5] // 这里还是 3 ?

那是因为,Go 会在 range 循环中自动为遍历的对象创造一个副本,可以理解为一个值拷贝,如果真的想修改原数组,你得这样写:

func main() {
   
    var m = []int{
   1, 3, 5}
    for i, v := range m {
   
        if i == 1 {
   
            m[i] = 2 // 改成 m[i]
        }
        fmt.Println(i, v)
    }
    fmt.Println(m)
}

程序输出是这样:

0 1
1 3 // 由于是值拷贝,所以改了原来的 m[i] 不影响 v 的值,m[i]=2,v=3.
2 5
[1 2 5] // 可以看到已经改成 2 了

也不算是什么大坑,理解了感觉和 PHP 的 foreach 语法差不多。


文章来源于本人博客,发布于 2019-06-16,原文链接:https://imlht.com/archives/187/

目录
相关文章
|
6天前
|
安全 网络协议 Go
Go语言网络编程
【10月更文挑战第28天】Go语言网络编程
94 65
|
6天前
|
网络协议 安全 Go
Go语言进行网络编程可以通过**使用TCP/IP协议栈、并发模型、HTTP协议等**方式
【10月更文挑战第28天】Go语言进行网络编程可以通过**使用TCP/IP协议栈、并发模型、HTTP协议等**方式
28 13
|
2天前
|
测试技术 Go
go语言中测试工具
【10月更文挑战第22天】
10 4
|
2天前
|
SQL 关系型数据库 MySQL
go语言中数据库操作
【10月更文挑战第22天】
12 4
|
2天前
|
缓存 前端开发 中间件
go语言中Web框架
【10月更文挑战第22天】
13 4
|
6天前
|
网络协议 安全 Go
Go语言的网络编程基础
【10月更文挑战第28天】Go语言的网络编程基础
23 8
|
5天前
|
Go
go语言的复数常量
【10月更文挑战第21天】
18 6
|
5天前
|
Go
go语言的浮点型常量
【10月更文挑战第21天】
13 4
|
5天前
|
编译器 Go
go语言的整型常量
【10月更文挑战第21天】
16 3
|
6天前
|
Go
go语言编译时常量表达式
【10月更文挑战第20天】
16 3

相关实验场景

更多