lambda表达式编译测试

我不会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如下:

image

也就是说这时候,这个lambda编译成了一个私有类的实例方法,而且数据就是外部被使用的变量

另外,每次调用TestLambda获取到的action.Target都是一个新的对象

原文地址:https://www.cnblogs.com/zgynhqf/p/1632086.html