C#5.0中捕获的闭包(循环变量)

时间:2013-04-28 15:12:35

标签: c# for-loop foreach closures c#-5.0

这在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循环的问题)?

1 个答案:

答案 0 :(得分:43)

  

这背后的原因是什么?

我会假设你的意思是“为什么for循环也没有改变?”

答案是,对于for循环,现有行为非常有意义。如果您将for循环中断为:

  • 初始化
  • 条件
  • 迭代

...然后循环大致:

{
    initializer;
    while (condition)
    {
        body;
        iterator;
    }
}

(当然,除了iteratorcontinue;语句的末尾执行外。

初始化部分逻辑上只发生一次,因此只有一个“变量实例化”完全符合逻辑。此外,在循环的每次迭代中,变量没有自然的“初始”值 - 没有什么可以说for循环具有是在初始化程序中声明变量的形式,在条件中测试它并在迭代器中修改它。你会期望像这样的循环:

for (int i = 0, j = 10; i < j; i++)
{
    if (someCondition)
    {
        j++;
    }
    actions.Add(() => Console.WriteLine(i, j));
}

将其与看起来foreach循环进行比较,就像你为每次迭代声明一个单独的变量一样。哎呀,变量是只读的,甚至可以认为它是一个在迭代之间变化的变量。 more 奇数。将foreach循环视为在每次迭代中声明一个新的只读变量,其值取自迭代器,这是完全合理的。