0

0

Go语言并发编程:解决Goroutine中循环变量捕获的常见问题

花韻仙語

花韻仙語

发布时间:2025-10-14 13:10:46

|

566人浏览过

|

来源于php中文网

原创

Go语言并发编程:解决Goroutine中循环变量捕获的常见问题

本文深入探讨了go语言中在使用goroutine和循环时常见的变量捕获陷阱。当goroutine在循环内部创建时,如果闭包直接引用循环变量,它们会捕获变量的引用而非其当时的值,导致所有goroutine最终都使用循环结束时的变量值。文章提供了详细的问题分析、正确的解决方案(通过参数传递变量副本)及跨语言对比,旨在帮助开发者避免此类并发编程错误。

问题描述

在Go语言中,当我们在循环内部启动Goroutine并让其访问循环变量时,经常会遇到一个出人意料的结果:所有Goroutine打印的都是循环变量的最终值,而不是它们在Goroutine创建时所期望的值。

考虑以下Go语言代码示例:

package main

import "fmt"

func main() {
  completed := make(chan bool, 2)
  m := map[string]string{"a": "a", "b": "b"}
  for k, v := range m {
    go func() {
      fmt.Println(k, v)
      completed <- true
    }()
  }
  <- completed
  <- completed
}

这段代码尝试遍历一个map,并为每个键值对启动一个Goroutine来打印它们。然而,实际运行结果往往是:

b b
b b

或者在某些情况下可能是 a a a a,但极少会出现 a a 和 b b 同时打印的情况。这让许多初学者感到困惑,误以为是某种奇怪的并发问题。

立即学习go语言免费学习笔记(深入)”;

原因分析:闭包与变量捕获

这种行为并非Go语言特有的并发问题,而是与编程语言中“闭包”如何捕获外部变量的机制有关。在Go语言中,匿名函数(即闭包)会捕获其定义时所在作用域的变量。当这些变量在循环中被声明和更新时,闭包捕获的是变量本身的“引用”或“内存地址”,而不是该变量在特定循环迭代时的“值”。

具体到上述示例:

  1. for k, v := range m 循环在每次迭代时会更新 k 和 v 这两个变量的值。
  2. go func() { ... }() 启动的匿名函数形成一个闭包,它捕获了外部作用域中的 k 和 v。
  3. 由于Goroutine的执行是异步的,通常情况下,当Goroutine真正开始执行 fmt.Println(k, v) 时,for 循环很可能已经完成了所有迭代,或者已经进行到后续的迭代。
  4. 此时,k 和 v 变量已经包含了循环的最终值(例如,map中最后一个元素的键和值)。因此,所有捕获了 k 和 v 的Goroutine都会读取到这些最终值,导致输出重复。

这与多线程无关,即使在单线程环境中,如果存在异步执行(如JavaScript中的 setTimeout),也会出现类似的问题。例如在JavaScript中:

obj = {a: 'a', b: 'b'};
for (k in obj) {
  setTimeout(function() { console.log(k, obj[k]); }, 0);
}

这段JavaScript代码同样会打印 b b 两次,因为 setTimeout 中的回调函数在执行时,for 循环已经结束,k 变量已经固定为 'b'。

Kuwebs企业网站管理系统3.1.5 UTF8
Kuwebs企业网站管理系统3.1.5 UTF8

酷纬企业网站管理系统Kuwebs是酷纬信息开发的为企业网站提供解决方案而开发的营销型网站系统。在线留言模块、常见问题模块、友情链接模块。前台采用DIV+CSS,遵循SEO标准。 1.支持中文、英文两种版本,后台可以在不同的环境下编辑中英文。 3.程序和界面分离,提供通用的PHP标准语法字段供前台调用,可以为不同的页面设置不同的风格。 5.支持google地图生成、自定义标题、自定义关键词、自定义描

下载

解决方案:通过参数传递变量副本

解决这个问题的关键在于确保每个Goroutine都拥有其自己独立的 k 和 v 值副本,而不是共享循环变量的引用。最常见的做法是将循环变量作为参数传递给Goroutine启动的匿名函数。

修改后的Go语言代码如下:

package main

import "fmt"

func main() {
  completed := make(chan bool, 2)
  m := map[string]string{"a": "a", "b": "b"}
  for k, v := range m {
    // 将 k 和 v 作为参数传递给匿名函数
    go func(key, value string) {
      fmt.Println(key, value)
      completed <- true
    }(k, v) // 在这里立即调用匿名函数,并传入当前迭代的 k 和 v 的值
  }
  <- completed
  <- completed
}

在这个修正后的代码中:

  1. go func(key, value string) { ... }(k, v) 这一行是核心。
  2. 在 go func(...) 之后紧跟着的 (k, v) 表示立即调用这个匿名函数,并将当前循环迭代中 k 和 v 的作为参数传递给它。
  3. 匿名函数的形参 key 和 value 会接收到这些值。由于 key 和 value 是匿名函数内部的局部变量,每个Goroutine都会拥有自己独立的 key 和 value 副本,它们与外部循环的 k 和 v 变量是完全独立的。
  4. 因此,当Goroutine执行时,它会打印出在它创建时捕获到的正确键值对。

运行修正后的代码,你将看到预期的输出:

a a
b b

或者

b b
a a

(顺序不确定,因为Goroutine的执行顺序是非确定性的)。

注意事项与最佳实践

  • go run -race 工具: Go语言提供了一个强大的数据竞争检测工具。如果你运行原始的错误代码,并使用 go run -race your_program.go 命令,它很可能会报告一个数据竞争(data race),因为多个Goroutine在读取 k 和 v 的同时,主Goroutine可能还在修改它们。这有助于发现这类潜在的问题。
  • 不仅仅是 for range: 这种变量捕获问题不仅限于 for range 循环,任何在循环内部创建闭包并引用循环变量的场景都可能遇到。例如,使用传统的 for i := 0; i
  • 创建局部变量副本: 除了通过参数传递,另一种常见的做法是在循环内部显式地创建循环变量的局部副本:
    for k, v := range m {
        kCopy := k // 创建 k 的局部副本
        vCopy := v // 创建 v 的局部副本
        go func() {
            fmt.Println(kCopy, vCopy)
            completed <- true
        }()
    }

    这种方式同样有效,因为它确保了Goroutine捕获的是每次迭代时 kCopy 和 vCopy 的独立引用,而不是外部循环的 k 和 v。

总结

在Go语言中,当在循环内部启动Goroutine时,理解闭包如何捕获循环变量至关重要。直接引用循环变量会导致所有Goroutine看到变量的最终值,而不是迭代时的特定值。通过将循环变量作为参数传递给Goroutine函数,或者在循环内部创建局部变量副本,可以有效地解决这个问题,确保每个Goroutine都处理其预期的独立数据。掌握这一技巧是编写健壮、可预测的Go并发程序的关键一步。

相关专题

更多
js获取数组长度的方法
js获取数组长度的方法

在js中,可以利用array对象的length属性来获取数组长度,该属性可设置或返回数组中元素的数目,只需要使用“array.length”语句即可返回表示数组对象的元素个数的数值,也就是长度值。php中文网还提供JavaScript数组的相关下载、相关课程等内容,供大家免费下载使用。

557

2023.06.20

js刷新当前页面
js刷新当前页面

js刷新当前页面的方法:1、reload方法,该方法强迫浏览器刷新当前页面,语法为“location.reload([bForceGet]) ”;2、replace方法,该方法通过指定URL替换当前缓存在历史里(客户端)的项目,因此当使用replace方法之后,不能通过“前进”和“后退”来访问已经被替换的URL,语法为“location.replace(URL) ”。php中文网为大家带来了js刷新当前页面的相关知识、以及相关文章等内容

374

2023.07.04

js四舍五入
js四舍五入

js四舍五入的方法:1、tofixed方法,可把 Number 四舍五入为指定小数位数的数字;2、round() 方法,可把一个数字舍入为最接近的整数。php中文网为大家带来了js四舍五入的相关知识、以及相关文章等内容

754

2023.07.04

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

478

2023.09.01

JavaScript转义字符
JavaScript转义字符

JavaScript中的转义字符是反斜杠和引号,可以在字符串中表示特殊字符或改变字符的含义。本专题为大家提供转义字符相关的文章、下载、课程内容,供大家免费下载体验。

454

2023.09.04

js生成随机数的方法
js生成随机数的方法

js生成随机数的方法有:1、使用random函数生成0-1之间的随机数;2、使用random函数和特定范围来生成随机整数;3、使用random函数和round函数生成0-99之间的随机整数;4、使用random函数和其他函数生成更复杂的随机数;5、使用random函数和其他函数生成范围内的随机小数;6、使用random函数和其他函数生成范围内的随机整数或小数。

1031

2023.09.04

如何启用JavaScript
如何启用JavaScript

JavaScript启用方法有内联脚本、内部脚本、外部脚本和异步加载。详细介绍:1、内联脚本是将JavaScript代码直接嵌入到HTML标签中;2、内部脚本是将JavaScript代码放置在HTML文件的`<script>`标签中;3、外部脚本是将JavaScript代码放置在一个独立的文件;4、外部脚本是将JavaScript代码放置在一个独立的文件。

658

2023.09.12

Js中Symbol类详解
Js中Symbol类详解

javascript中的Symbol数据类型是一种基本数据类型,用于表示独一无二的值。Symbol的特点:1、独一无二,每个Symbol值都是唯一的,不会与其他任何值相等;2、不可变性,Symbol值一旦创建,就不能修改或者重新赋值;3、隐藏性,Symbol值不会被隐式转换为其他类型;4、无法枚举,Symbol值作为对象的属性名时,默认是不可枚举的。

553

2023.09.20

Java编译相关教程合集
Java编译相关教程合集

本专题整合了Java编译相关教程,阅读专题下面的文章了解更多详细内容。

9

2026.01.21

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
React 教程
React 教程

共58课时 | 3.9万人学习

TypeScript 教程
TypeScript 教程

共19课时 | 2.3万人学习

Bootstrap 5教程
Bootstrap 5教程

共46课时 | 2.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号