这个问题是在最近一次英格兰Brighton ALT.NET Beers活动中提出来的。我发现,如果不用代码来演示,你很难单用话语把它解释清楚,所以,在这里,我打算用C#来解释一下什么是闭包(closures)。维基百科上说:
在计算机科学中,闭包(Closure)是词法闭包(Lexical Closure)的简称,是引用了自由变量的函数。这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外。所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合而成的实体。
所以,一个闭包就是一个“捕获”或“携带”了其被生成的环境中、所属的变量范围内所引用的所有变量的函数。的确,很难描述,但当你看完了这些代码后,你就很容易理解了。
var x = 1; Action action = () => { var y = 2; var result = x + y; Console.Out.WriteLine("result = {0}", result); }; action();
这里我们首先定义了一个变量“x”,值为1。然后我们定义了一个匿名函数(一个lambda表达式)赋给类型Action。Action没有参数,没有返回值,但如果你观察“action”里的定义,你会发现它使用了“x”变量。这是变量是被action“捕获”或“携带”的,自动被添加到了action的运行环境中了。
当我们执行action时,它输出了我们预期的结果。请注意,当我们执行时,原始的“x”此时已经脱离了它当初的变量环境,但它仍然能用。
当你在代码调试器(debugger)里观察“action”时,会发现很有趣的事情。我们可以看到,C#编译器为我们创建了一个Target类,里面封装了x变量:
闭包(和higher order functions)都是非常有用的东西。如果你曾经开发过稍微复杂一点的Javascript程序,你可能就会知道,这个东西可以被当成很多面向对象特征的替代品,就像C#那样。前不久我还在C#里写了一个例子来验证这种想法。
习惯性的,John Skeet给了闭包一个更详细的描述。更详细的信息请查看《深入解析C#》里的这一章。里面还介绍了你在闭包里会经常会遇到的一些错误。
原文评论:
huansinho:我out了。。。C#有闭包。。。 - 2011年08月13日3:49 上午
泥菩萨:我也是第一次听说。哎。 - 2011年08月13日1:12 下午
bill:当我们执行acton action打错了~~ - 2011年08月13日7:18 下午
Aqee:谢谢提醒,已改正 - 2011年08月15日9:39 下午
fezhang 说:
2011年08月13日9:43 下午
closure就是把上下文不放到自己域, 比如说一些变量来自稍大或更大的域,但只有运行时才绑定 — 这给闭包函数的编写者带来不小想象空间.
dytes 说:
2011年08月18日9:45 下午
闭包是由函数和与其相关的引用环境组合而成的实体。算是见到的最简明扼要的闭包解释了。
chris.wu 说:
2011年08月19日11:58 上午
真得不错,解释的到位,代码很贴切,不是那么抽象
“闭包是由函数和与其相关的引用环境组合而成的实体。算是见到的最简明扼要的闭包解释了”
转自:http://www.aqee.net/what-is-a-closure/
[本文英文原文链接:What is a Closure? ]