我不会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都是一个新的对象。