技术文摘
Go匿名函数变量捕获:闭包中变量i为何永远是4
2025-01-09 02:07:44 小编
Go匿名函数变量捕获:闭包中变量i为何永远是4
在Go语言的编程世界里,匿名函数与闭包是强大且常用的特性,但它们也带来了一些容易让人困惑的问题,其中一个典型的现象就是:闭包中捕获的变量i为何在循环结束后永远是4 。下面让我们深入探讨一下这个问题。
我们来看一段简单的代码示例:
package main
import (
"fmt"
)
func main() {
var funcs []func()
for i := 0; i < 4; i++ {
funcs = append(funcs, func() {
fmt.Println(i)
})
}
for _, f := range funcs {
f()
}
}
你可能预期这段代码会依次输出0、1、2、3,但实际运行结果却是4、4、4、4 。这背后的原因在于Go语言中闭包捕获变量的机制。
在Go中,匿名函数捕获的是变量的引用,而非值。在上述循环中,变量i只有一个实例,每次迭代时,i的值会不断更新。当我们将匿名函数添加到funcs切片中时,这些匿名函数捕获的都是同一个变量i的引用。
直到循环结束,i的值最终变成了4 。之后,当我们遍历funcs切片并调用这些匿名函数时,它们所访问的i已经是最终的值4 。
要想实现预期的输出0、1、2、3 ,有几种解决方案。一种方法是通过传参的方式,将i的值作为参数传递给匿名函数,因为函数参数是值传递:
package main
import (
"fmt"
)
func main() {
var funcs []func()
for i := 0; i < 4; i++ {
temp := i
funcs = append(funcs, func() {
fmt.Println(temp)
})
}
for _, f := range funcs {
f()
}
}
在这个修改后的代码中,每次循环创建一个临时变量temp ,它复制了i当前的值。匿名函数捕获的是temp的引用,而每个temp的值都是独立的,所以在调用匿名函数时,就能输出正确的结果。
理解Go语言中匿名函数变量捕获机制对于编写正确且高效的代码至关重要。掌握这些细节,能帮助我们避免这类常见的错误,充分发挥闭包的强大功能。
- Mock 框架的三次迭代助力高效单元测试
- 高并发秒杀系统之总结
- TOP5 机器学习框架在 Web 开发中的应用盘点
- Docker 在雪球的技术实践:容器的正确使用方式
- 京东架构师解读系统练级攻略
- A/B 测试需知的五个要点
- 4 个实用的微服务测试策略推荐
- 基于自身数据构建简易卷积神经网络
- 谷歌云与 Unity 开源 Open Match 玩家配对方案
- RPC 系统的设计方法
- Python 学习的五大心动理由
- JavaScript 中实现简易 Vue 的方法
- JavaScript 任务表的十五种实用功能
- 未被察觉的隐患:危险的 target="_blank" 与 “opener”
- CSS 水平垂直居中的 1010 种实现方法(终极汇总)