从本质上说,闭包是一段可以在晚些时候执行的代码块,但是这段代码块依然维护着它第一个被创建时环境(执行上下文)- 即它仍可以使用创建它的方法中局部变量,即使那个方法已经执行完了。C#中通常通过匿名函数和lamada表达式来实现闭包
然而,当在循环中使用闭包时,需要小心变量捕获的问题。这是因为闭包捕获的是变量的引用,而不是变量的当前值。
using System;
using System.Collections.Generic;
class Program
{
static void Main()
{
List<Action> actions = new List<Action>();
for (int i = 0; i < 5; i++)
{
actions.Add(() => Console.WriteLine(i));
}
foreach (var action in actions)
{
action();
}
}
}
期望上面的代码输出:01234 但实际输出是55555
要解决这个问题,可以在循环内部创建一个新的变量,使得每个闭包捕获的是不同的变量。
using System;
using System.Collections.Generic;
class Program
{
static void Main()
{
List<Action> actions = new List<Action>();
for (int i = 0; i < 5; i++)
{
int local = i; // 声明一个新的变量
actions.Add(() => Console.WriteLine(local));
}
foreach (var action in actions)
{
action();
}
}
}