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循环变量是相对的全局变量,这也是导致闭包陷阱的原因
这些知识点都很重要且较为晦涩,建议读者关注文中给出的参考资料,以便进一步学习和探讨。
参考资料
热门推荐
厌氧胶的使用方法及注意事项
龙虾凭什么这么贵,难道只是因为大吗?
留学生视角:83条创意龙虾推广语的灵感碰撞
朋友小聚文案,让你秒变朋友圈人气王!
哈尔滨传统文化大盘点:从冰雕艺术到东北大秧歌
哈尔滨领事馆老建筑:冬日打卡的“国际范儿”
哈尔滨冰雪节:铁锅炖大鹅的诱惑
如何做好薪酬福利体系设计,更好实现员工激励?
怎么注销营业执照?没有营业执照招聘违法吗?生产洗洁精需要许可证吗?
《谁是卧底》再登顶:聚会必备神作,考验智商与情商的社交游戏
胖姐生日聚会:一场维系女生友谊的仪式
广州帽峰山生态园:好友聚会的完美一天
红烧狮子头:家庭聚会的创意料理首选
北大教授揭秘:大眼鱼龙如何从陆地霸主变身海洋巨兽
玛丽·安宁:开启古生物学新纪元的化石猎人
如何管理基建项目
一篇文章搞懂中国古代的姓、氏、名、字、号
涨停板股票难买?掌握这些技巧就能提高成功率!
如何看待封涨停现象并把握投资机会?这种投资机会的把握存在哪些风险?
西安《长恨歌》:一场跨越千年的爱情传奇
警惕假布洛芬!教你识别正规渠道
布洛芬和戴芬,谁更适合你?
冬季流感高发,如何正确使用布洛芬?
双十一选购170L尺码衣服攻略
亲子关系的界限与自由:如何在关爱与独立之间找到平衡
“边界感”让亲子关系更舒适
长三角档案便民服务升级:出生医学证明档案可在线查询
出生证明办理流程:新生儿的第一份官方文件
如何准备父子关系证明文件?这些文件在法律上的作用如何?
哈尔滨冰雪大世界:今年新玩法提前揭秘!