这在C#5.0中工作正常(按预期方式):
var actions = new List<Action>();
foreach (var i in Enumerable.Range(0, 10))
{
actions.Add(() => Console.WriteLine(i));
}
foreach (var act in actions) act();
打印0到9.但是这个显示10次10次:
var actions = new List<Action>();
for (var i = 0; i < 10; i++)
{
actions.Add(() => Console.WriteLine(i));
}
foreach (var act in actions) act();
问题:这是我们在5.0之前的C#版本中遇到的问题;所以我们不得不使用一个循环局部占位符来进行闭包,它现在已经修复了 - 在C#5.0中 - 在“foreach”循环中。但不是“for”循环!
这背后的原因是什么(不解决for
循环的问题)?
答案 0 :(得分:43)
这背后的原因是什么?
我会假设你的意思是“为什么for
循环也没有改变?”
答案是,对于for
循环,现有行为非常有意义。如果您将for
循环中断为:
...然后循环大致:
{
initializer;
while (condition)
{
body;
iterator;
}
}
(当然,除了iterator
在continue;
语句的末尾执行外。
初始化部分逻辑上只发生一次,因此只有一个“变量实例化”完全符合逻辑。此外,在循环的每次迭代中,变量没有自然的“初始”值 - 没有什么可以说for
循环具有是在初始化程序中声明变量的形式,在条件中测试它并在迭代器中修改它。你会期望像这样的循环:
for (int i = 0, j = 10; i < j; i++)
{
if (someCondition)
{
j++;
}
actions.Add(() => Console.WriteLine(i, j));
}
将其与看起来的foreach
循环进行比较,就像你为每次迭代声明一个单独的变量一样。哎呀,变量是只读的,甚至可以认为它是一个在迭代之间变化的变量。 more 奇数。将foreach
循环视为在每次迭代中声明一个新的只读变量,其值取自迭代器,这是完全合理的。