关于Golang中for-loop与goroutine的问题详解

作者:寇池滨 时间:2024-04-29 13:03:16 

背景

最近在学习MIT的分布式课程6.824的过程中,使用Go实现Raft协议时遇到了一些问题。分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧。

参见如下代码:


for i := 0; i < len(rf.peers); i++ {
 DPrintf("i = %d", i)
if i == rf.me {
  DPrintf("skipping myself #%d", rf.me)
  continue
 }
go func() {
  DPrintf("len of rf.peers = %d", len(rf.peers))
  DPrintf("server #%d sending request vote to server %d", rf.me, i)
  reply := &RequestVoteReply{}
  ok := rf.sendRequestVote(i, args, reply)
  if ok && reply.VoteGranted && reply.Term == rf.currentTerm {
   rf.voteCount++
   if rf.voteCount > len(rf.peers)/2 {
    rf.winElectionCh <- true
   }
  }
 }()
}

其中,peers切片的长度为3,因此最高下标为2,在非并行编程中代码中的for-loop应该是很直观的,我当时并没有意识到有什么问题。可是在调试过程中,一直在报 index out of bounds 错误。调试信息显示i的值为3,当时就一直想不明白循环条件明明是 i < 2,怎么会变成3呢。

分析

虽然不明白发生了什么,但知道应该是循环中引入的 goroutine 导致的。经过Google,发现Go的wiki中就有一个页面 Common Mistake - Using goroutines on loop iterator variables 专门提到了这个问题,看来真的是很 common 啊,笑哭~

初学者经常会使用如下代码来并行处理数据:


for val := range values {
go val.MyMethod()
}

或者使用闭包(closure):


for val := range values {
go func() {
 fmt.Println(val)
}()
}

这里的问题在于 val 实际上是一个遍历了切片中所有数据的单一变量。由于闭包只是绑定到这个 val 变量上,因此极有可能上面的代码的运行结果是所有 goroutine 都输出了切片的最后一个元素。这是因为很有可能当 for-loop 执行完之后 goroutine 才开始执行,这个时候 val 的值指向切片中最后一个元素。


The val variable in the above loops is actually a single variable that takes on the value of each slice element. Because the closures are all only bound to that one variable, there is a very good chance that when you run this code you will see the last element printed for every iteration instead of each value in sequence, because the goroutines will probably not begin executing until after the loop.

解决方法

以上代码正确的写法为:


for val := range values {
go func(val interface{}) {
 fmt.Println(val)
}(val)
}

在这里将 val 作为一个参数传入 goroutine 中,每个 val 都会被独立计算并保存到 goroutine 的栈中,从而得到预期的结果。

另一种方法是在循环内定义新的变量,由于在循环内定义的变量在循环遍历的过程中是不共享的,因此也可以达到同样的效果:


for i := range valslice {
val := valslice[i]
go func() {
 fmt.Println(val)
}()
}

对于文章开头提到的那个问题,最简单的解决方案就是在循环内加一个临时变量,并将后面 goroutine 内的 i 都替换为这个临时变量即可:


server := i

来源:https://segmentfault.com/a/1190000010884717

标签:golang,for-loop,goroutine
0
投稿

猜你喜欢

  • 后台程序开发常用jQuery插件

    2010-04-06 12:11:00
  • 对python3标准库httpclient的使用详解

    2021-09-07 06:48:02
  • Python利用for循环打印星号三角形的案例

    2022-09-20 22:22:00
  • 用Python分析3天破10亿的《我不是药神》到底神在哪?

    2023-04-29 06:16:26
  • Python实现嵌套列表及字典并按某一元素去重复功能示例

    2023-02-22 10:44:05
  • python定义具名元组实例操作

    2023-05-16 22:22:55
  • Python readline()和readlines()函数实现按行读取文件

    2022-02-21 13:19:36
  • Go1.18新特性对泛型支持详解

    2024-05-22 17:46:25
  • 微信小程序实现登陆注册滑块验证

    2023-08-24 17:36:26
  • Django自定义插件实现网站登录验证码功能

    2022-07-13 12:31:13
  • python四个坐标点对图片区域最小外接矩形进行裁剪

    2022-01-18 02:18:09
  • 分享11个Python自动化操作Excel的方法

    2022-11-18 08:10:39
  • 深入了解Golang 哈希算法之MD5、SHA-1和SHA-256

    2024-05-05 09:34:51
  • python基于paramiko将文件上传到服务器代码实现

    2022-05-20 06:14:44
  • django2.0扩展用户字段示例

    2023-08-31 11:10:10
  • MySQL定时任务(EVENT事件)如何配置详解

    2024-01-19 08:04:12
  • 深入理解python对json的操作总结

    2022-04-08 20:36:01
  • python的字典和集合你了解吗

    2022-12-13 11:28:44
  • 使用python-cv2实现Harr+Adaboost人脸识别的示例

    2022-03-16 01:05:10
  • python实现控制电脑鼠标和键盘,登录QQ的方法示例

    2023-11-19 12:10:26
  • asp之家 网络编程 m.aspxhome.com