使用 Closures 改善 API 的设计与使用

  • Jonathan Allen
  • 张逸

2008 年 9 月 16 日

话题:.NET函数式编程C#语言 & 开发架构

某些执行复杂解析的 API 通常会通过事件暴露中间结果。例如,XDocument.Validate 扩展方法使用了一个事件来通知程序存在的验证错误。

在传统的.NET 编程中,捕获这些事件就意味着需要为每个事件创建一个单独的函数。由于没有直接的途径传递事件处理器的额外信息,通常这些函数会在抛出事件的类中来实现。

Eric White 使用 Closures 演示了一种更好的方式。Closures 在 lambda 或匿名内联函数中,属于一种特殊情形。其特殊之处在于能够引用一种变量,这种变量并非直接赋值,而是赋给包含了 lambda 的函数。这就使得 Eric 可以将事件处理器的逻辑赋值给函数。如下的代码行演示了创建 Closure,并将其传递给 Validate 函数的方法:

snippet.Validate(schemas, (o, e) => errors += e.Message + Environment.NewLine);

正如在批注中所示,C# 和 VB 中的 Closure 均被实现为匿名类,它包含了必要的成员变量。“被装入”的变量被移到新类中,既可通过原有函数,也可以通过包含它的任意匿名函数所引用。

查看英文原文:Using Closures to Improve API Design and Usage

Faisal Waris 的精彩评论: 如果使用 Closures,则程序的某些部分会变得更为简洁。正如在一些旧有语言中开一时风气之先的语言那样(LISP、Smalltalk 等),大多数新的主流语言都支持 Closures(VB、C#、Ruby、Python、javascript)。Java 是一个值得注意的例外 [译注:在 Java 7 中引入了对 Closures 的支持]。然而并非所有人(特别是 C#/VB 社区)都对此有所了解,或者欣然接受,因而对其给与更多的关注总是一件好事。值得注意的是,异步编程从 Closures 中获益良多——最显而易见的证据就是 AJAX 对其的广泛使用。Closures 在 DLR 中的支持,对 Silverlight 编程无疑起到了推波助澜的作用,因为所有的服务端调用都是在调用体之外进行异步处理的。

.NET函数式编程C#语言 & 开发架构