我不会IL,也不想学IL,所以为了测试Lambda表达式编译的原理,只能使用调试进行试验来认识:
第一种情况,当Lambda表达式没有用到函数外的变量时:
class MyClass2
{
public Action TestLambda()
{
Action action = () =>
{
Console.WriteLine(DateTime.Now);
};
return action;
}
public static void Test()
{
MyClass2 c2 = new MyClass2();
var action = c2.TestLambda();
var action2 = c2.TestLambda();
Console.WriteLine(action.Method.DeclaringType.FullName);//"ConsoleApp.Program+MyClass2"
Console.WriteLine(action.Method.IsStatic);//"true"
Console.WriteLine(action.Target);//"" null
Console.WriteLine(action == action2);//true
}
}
这种情况下,action被编译成了MyClass2的静态方法。
第二种情况,当Lambda表达式使用到函数外的变量时:
class MyClass
{
public Action TestLambda()
{
int i = 1;
int j = 5;
Action action = () =>
{
Console.WriteLine(i);
Console.WriteLine(j);
};
i++;
return action;
}
public static void Test()
{
MyClass c = new MyClass();
var action = c.TestLambda();
var action2 = c.TestLambda();
action();//2
Console.WriteLine(action.Target.GetType().FullName);//"ConsoleApp.Program+MyClass+<>c__DisplayClass6"
Console.WriteLine(action.Method.DeclaringType.FullName);//"ConsoleApp.Program+MyClass+<>c__DisplayClass6"
Console.WriteLine(action == action2);//false
Console.WriteLine(action.Target == action2.Target);//false
}
}
调试发现action.Target如下:

也就是说这时候,这个lambda编译成了一个私有类的实例方法,而且数据就是外部被使用的变量。
另外,每次调用TestLambda获取到的action.Target都是一个新的对象。
本文转自BloodyAngel博客园博客,原文链接:http://www.cnblogs.com/zgynhqf/archive/2009/12/25/1632086.html,如需转载请自行联系原作者