我一直在学习C#,并试图理解lambdas。在下面的示例中,它打印了10次。
class Program
{
delegate void Action();
static void Main(string[] args)
{
List<Action> actions = new List<Action>();
for (int i = 0; i < 10; ++i )
actions.Add(()=>Console.WriteLine(i));
foreach (Action a in actions)
a();
}
}
显然,lambda后面生成的类存储了一个指向int i
变量的引用或指针,并在每次循环迭代时为同一引用分配一个新值。有没有一种方法可以强制lamda获取一个副本,比如C++0x语法
[&](){ ... } // Capture by reference
与
[=](){ ... } // Capture copies
发布于 2009-01-16 20:24:23
编译器所做的是将您的lambda和lambda捕获的任何变量拉入编译器生成的嵌套类中。
编译后,您的示例看起来很像这样:
class Program
{
delegate void Action();
static void Main(string[] args)
{
List<Action> actions = new List<Action>();
DisplayClass1 displayClass1 = new DisplayClass1();
for (displayClass1.i = 0; displayClass1.i < 10; ++displayClass1.i )
actions.Add(new Action(displayClass1.Lambda));
foreach (Action a in actions)
a();
}
class DisplayClass1
{
int i;
void Lambda()
{
Console.WriteLine(i);
}
}
}
通过在for循环中创建一个副本,编译器在每次迭代中生成新对象,如下所示:
for (int i = 0; i < 10; ++i)
{
DisplayClass1 displayClass1 = new DisplayClass1();
displayClass1.i = i;
actions.Add(new Action(displayClass1.Lambda));
}
发布于 2009-01-16 20:08:57
我能找到的唯一解决方案是先制作一个本地副本:
for (int i = 0; i < 10; ++i)
{
int copy = i;
actions.Add(() => Console.WriteLine(copy));
}
但是我很难理解为什么在for循环中放置一个副本与使用lambda捕获i
有什么不同。
发布于 2009-01-16 20:13:06
唯一的解决方案是创建一个本地副本,并在lambda中引用它。当在闭包中访问C# (和VB.Net)中的所有变量时,它们将具有引用语义与复制/值语义。在这两种语言中都无法更改此行为。
注意:它实际上不会作为引用进行编译。编译器将变量提升到闭包类中,并将对"i“的访问重定向到给定闭包类中的字段"i”。不过,通常更容易将其视为引用语义。
https://stackoverflow.com/questions/451779
复制相似问题