[轉載] Closure 翻譯文章

因為我連不到原中文翻譯的網址了,所以只好把找到的文章轉貼過來。Closure 和 Block 是 Ruby 中很重要的特色,而 Martin Fowler 的這篇則用 Ruby 來舉例解釋 Closure。

Ruby提供了好用的 syntax 跟 invocation 來支援 closure。是的,Syntax matters

原著:Martin Fowler martinfowler.com/bliki/Closures.html

翻譯:liubin (at) huangpuzhuang.com

另外一片文章總結了各種語言實現的本文中的例子。閉包(Closures)在各種語言中的例子

2004/11/23

本文地址:www.ruby-cn.org/articles/closures.html

隨著人們對動態語言興趣的日益濃厚,越來越多的人都遇到了閉包(Closures )和或塊(Blocks)等概念。有著C/C++/Java/C#等語言背景的人因為這些語言本身沒有閉包這個概念,所以可能不太了解閉包。本文將簡單的介紹一下閉包的概念,那些有大量支持閉包語言編程經驗的人也許覺得本文不會太有意思。閉包的概念已經提出很長時間了。我第一次碰到這它是在smalltalk中,那時候還叫做塊(blocks)。Lisp語言中用的很多。Ruby中也有同樣的功能-這也是Ruby用戶喜歡Ruby的一個原因。

本質上來說,一個閉包是一塊代碼,它們能作為參數傳遞給一個方法調用。我將通過一個簡單的例子來闡述這個觀點。假設我們有一個包含一些雇員對象的列表,然後我想列出職位為經理的員工,這樣的員工可以通過IsManager判斷。在C#裡,我們可能會寫出下麵類似的代碼:

public static IList Managers(IList emps) {
  IList result = new ArrayList();
  foreach(Employee e in emps)
   if (e.IsManager) result.Add(e);
    return result;
}

在一種支持閉包的語言中,比如Ruby,我們可以這樣寫:

def managers(emps)
  return emps.select {|e| e.isManager}
end

select是Ruby中定義的集合結構中的一個方法,它接受一個block,也就是閉包,作為一個參數。在Ruby中,閉包寫在一對大括號中(不止這一種方法,另一種為do .. end)。如果這個塊也接受參數,你可以將這些參數放到兩個豎線之間。select方法循環迭代給定的數組,對每個元素執行給定的block,然後將每次執行block返回true的元素組成一個新的數組再返回。

現在,如果你是C程序員你也許要想,通過函數指針也可以實現,如果你是JAVA程序員,你可能回想我可以用匿名內類來實現,而一個C#者則會想到代理(delegate)。這些機制和閉包類似,但是它們和閉包之間有兩個明顯得區別。

第一個是形式上的不同(The first one is a formal difference)。閉包可以引用它定義時候可見的變量。看看下面的方法:

def highPaid(emps)
 threshold = 150
 return emps.select {|e| e.salary > threshold}
end

注意select的block代碼中引用了在包含它的方法中的局部變量,而其它不支持真正閉包的語言使用其它方法達到類似功能的方法則不能這樣做。閉包還允許你做更有趣的事情,比如下面方法:

def paidMore(amount)
 return Proc.new {|e| e.salary > amount}
end

這個方法返回一個閉包,實際上它返回一個依賴於傳給它的參數的閉包。我可以用一個參數創建一個這樣的方法,然後再把它賦給另一個變量。

highPaid = paidMore(150)

變量 highPaid 包含了一段代碼(在Ruby中是一個Proc對象),這段代碼將判斷一個對象的salary屬性是否大於150。我們可以這樣使用這個方法:

john = Employee.new
john.salary = 200
print highPaid.call(john)

表達式highPaid.call(john)調用我之前定義的代碼,這時候此代碼中的amount已經在創建這方法的時候綁定為150。即使現在我執行print 的時候,150已經不在它的範圍內了,但是amount和150之間的綁定依然存在。

所以,閉包的第一個關鍵點是閉包是一段代碼加上和定義它的環境之間的綁定(they are a block of code plus the bindings to the environment they came from)。這是閉包和函數指針等其它相似技術的不同點(java匿名內類可以訪問局部變量,但是只有當這些內類是final的時候才行)。

第二個不同點不是定義形式的不同,但是也同樣重要。(The second difference is less of a defined formal difference, but is just as important, if not more so in practice)。支持閉包的語言允許你用很少的語法去定義一個閉包,儘管這點可能不是很重要的一點,但我相信這點是至關重要的-這是使得人們能很自然 的使用閉包的關鍵點。看看Lisp,Smalltalk和Ruby,閉包遍布各處-比其它語言中類似的使用多很多。綁定局部變量是它的特點之一,但我想最 大的原因是使用閉包的語法和符號非常簡單和清楚。

一個很好的相關例子是從Smalltalk程序員到JAVA程序員,開始時很多人,包括我,試驗性的將在Smalltalk中使用閉包的地方在Java中使用匿名內類來實現。但結果使得代碼變得混亂難看,所以我們不得不放棄。

我在Ruby經常使用閉包,但我不打算創建Proc對象,然後傳來傳去。大多數時間我用閉包來處理前面我提到的select等基於集合對象的方法。閉包另一個重要用途是’execute around method’,比如處理一個文件:

File.open(filename) {|f| doSomethingWithFile(f)}

這裡open方法打開一個文件,然後執行給定的block,然後關閉它。這樣處理非常方便,尤其是對事務(要求commit或者rollback),或者其它的你需要在處理結束時候作一些收尾處理的事情。我在我的xml文檔轉換中廣泛使用這個優點。

閉包的這些用法顯然遠不如用Lisp語言的人遇到的多,即使我,在使用沒有閉包支持的語言的時候,也會想念這些東西。閉包就像一些你第一眼見到覺得不怎麼樣的東西,但你很快就會喜歡上它們。

其它語言例子

Joe Walnes在blog中提供了 closures in the next version of C#。這個例子是靜態類型語言的,基於delegate,且需要delegate關鍵字。

更新: Ivan Moore提供類類似的 Python 的例子

更新: Vadim Nasardinov 讓我知道了來自Guy Steeleled的 closures in Java 這個有趣的珍聞。 

譯者註:如你想知道上面例子中文件對像是怎麼自己關閉的,請看blog.csdn.net/ruby_cn/archive/2004/11/23/192588.aspx,希望可以找到答案。翻譯的不好,請多原諒。歡迎交流。括號之中的英語實在是不知如何很好翻譯,所以保留了下來。

參與討論

1 則留言

  1. 自動引用通知: 闭包概念和应用场景

發佈留言

發表迴響