在《C#高级编程》一书中提到通过Lambda表达式可以访问Lambda表达式块外部的变量 ,这是一个很好的功能(类似Js中的 闭包)。但是如果没有正确的使用,会非常危险。

比如下面的事例中

int someVal  = 5;

Fun<int, int> f = x => x + someVal;

Console.WriteLine(f(3));

我们这个表达式的本意是返回一个数x+5的结果。所以f(3)的结果应该是 3 + someVal = 8。

但是如果以后在不经意中修改了someVal的值,则会出现我们意想不到的结果。

比如:

someVal = 7;

Console.WriteLine(f(3));

此时的打印的结果是10。

特别是当通过多个线程调用Lambda时,我们可能不知道此时someVal的值到底是多少,从而导致不可预知的结果,所以要慎重使用。

那么,在Lambda表达式中使用外部变量的原理是什么呢。

原来,在运行Lambda表达式时,编译器会创建一个匿名类,他可以通过构造函数来传递外部变量。该构造函数的参数取决于外部传递进来的变量个数。对于上面的表达式,匿名类如下所示:

public class AnonymousClass     {

    private int someVal;

 public AnonymousClass(int someVal)         {             this.someVal = someVal;         }

public int Anonymous(int x)         {             return x + someVal;         }

}

这样就可以理解为什么Lambda表达式可以使用外部变量了。

04-18 11:26