C# foreach语法糖背后的秘密:闭包与迭代变量的那些事
创作时间:
作者:
@小白创作中心
C# foreach语法糖背后的秘密:闭包与迭代变量的那些事
引用
1
来源
1.
https://cloud.tencent.com/developer/article/2196843
本文通过对比C#中的for循环和foreach循环在闭包处理上的差异,深入探讨了为什么foreach能够正确输出迭代值,而for循环则不能。文章还进一步扩展到Go语言的for循环陷阱,并提供了相应的解决方案。
让我们先来看一个C#的示例代码:
var t1 = new List<Action>();
for (int i = 0; i < 5; i++)
{
var func = (() =>
{
Console.WriteLine(i);
});
t1.Add(func);
}
foreach (var item in t1)
{
item();
}
左边输出5个5;右边输出0,1,2,3,4。
闭包原理
闭包是在词法环境中捕获自由变量的头等函数。在上述代码中,关键在于闭包捕获的自由变量。
这里有三个关键概念需要理解:
- 局部变量i是被头等函数引用的自由变量
- 闭包捕获变量i的时空和闭包执行的时空不是一个时空
- 所有闭包执行时,捕获的都是变量i的最终值
解决方案
为了应对这种陷阱,可以在循环内使用一个局部变量来解构每个闭包与全局自由变量i的关系:
var t1 = new List<Action>();
for (int i = 0; i < 5; i++)
{
var j = i;
var func = (() =>
{
Console.WriteLine(j);
});
t1.Add(func);
}
foreach (var item in t1)
{
item();
}
foreach的底层实现
foreach的底层实现依赖于IEnumerable和IEnumerator两个接口。下面是foreach的伪代码:
{
E e = ((C)(x)).GetEnumerator();
try
{
while (e.MoveNext())
{
V v = (V)(T)e.Current; // 注意,变量v的定义是在循环体内
/*embedded_statement*/
}
}
finally
{
... // Dispose e
}
}
由于变量v的定义在while循环内部,因此使用foreach迭代时,每个闭包捕获的都是局部的自由变量,因此foreach闭包执行能输出0,1,2,3,4。
Golang的for循环陷阱
Golang中只有for循环,没有while和foreach关键字。下面是一个示例:
package main
import "fmt"
var slice []func()
func main() {
sli := []int{1, 2, 3, 4, 5}
for _, v := range sli {
fmt.Println(&v, v)
slice = append(slice, func() {
fmt.Println(v)
})
}
for _, val := range slice {
val()
}
}
输出结果:
0xc00001c098 1
0xc00001c098 2
0xc00001c098 3
0xc00001c098 4
0xc00001c098 5
5
5
5
5
5
Golang的for-range循环本质上与C#的for循环相同,闭包引用的是全局的自由变量v。解决方法是在循环体内创建局部变量:
for _, v := range sli {
v := v
fmt.Println(&v, v)
slice = append(slice, func() {
fmt.Println(v)
})
}
总结
本文主要讨论了以下几个知识点:
- 闭包:是在词法环境中捕获自由变量的头等函数
- foreach语法糖:依赖于IEnumerable和IEnumerator接口实现,同时foreach每次迭代使用的是块内局部变量
- for循环变量是相对的全局变量,这也是导致闭包陷阱的原因
这些知识点都很重要且较为晦涩,建议读者关注文中给出的参考资料,以便进一步学习和探讨。
参考资料
热门推荐
资产负债表分析:一眼看穿企业财务状况的5个要点
企业培训讲课内容如何编写
三角梅为什么会卷叶,存在温度不适、光照太强、缺少水分等因素
柳宗元《晋问》:一篇赞美晋地山河与物产的文学佳作
如何巧妙地发群公告并吸引注意
从 Manus 的爆火看利用热词的引流技巧
数据分析行业:揭秘当今最炙手可热的职业领域
HDMI 2.0 是否支持 140Hz 刷新率?详细解析与兼容性探讨
HDMI 2.0和HDMI 2.1的区别
什么是人工智能(AI)?其在SaaS中的应用与未来趋势
细胞及其增殖、调控机制的发现历程
如何通过社保局官网查询个人账户缴费明细?
涨知识!贡菜到底是不是莴笋?还有哪些蔬菜是“亲戚”?
如何做好课前预习:七个方面的实用建议
揭秘!一日三餐如何科学摄入1200大卡
公司经营情况分析 - 了解你所投资公司的真实状况
智能体脂称硬件设计详解:从电源到数据传输的完整方案
从实验室看翡翠,如何鉴定真假?
深圳网站建设内容管理中的版权问题:如何避免侵权?
如何通过练习提升倒车技能?这些练习方法对驾驶安全有何影响?
如何选择合适的家居用品?这些用品有哪些潜在的优势和劣势?
江西省赣州市:加速“融湾” 展现高质量发展新势头
如何描写童年生活的地方?童年记事:如何生动描绘你儿时的乐园?!
无法连接服务器?这些原因和解决方法请收好
生命守护之门,带您揭开 ICU 的神秘面纱
学霸的高效预习法:三个基本原则让学习效率翻倍
Skype如何禁止陌生人私聊?
JWT揭秘:前后端安全与双Token策略全解析
企业人力资源管理系统的ER图包含哪些实体和关系?
如何勇于尝试新事物