<rt id="bn8ez"></rt>
<label id="bn8ez"></label>

  • <span id="bn8ez"></span>

    <label id="bn8ez"><meter id="bn8ez"></meter></label>

    ivaneeo's blog

    自由的力量,自由的生活。

      BlogJava :: 首頁 :: 聯系 :: 聚合  :: 管理
      669 Posts :: 0 Stories :: 64 Comments :: 0 Trackbacks
     隨著人們對動態語言興趣的日益濃厚,越來越多的人都遇到了閉包(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語言的人遇到的多,即使我,在使用沒有閉包支持的語言的時候,也會想念這些東西。閉包就像一些你第一眼見到覺得不怎么樣的東西,但你很快就會喜歡上它們。

    posted on 2005-08-23 17:06 ivaneeo 閱讀(254) 評論(0)  編輯  收藏 所屬分類: ruby-寶石也鋒芒
    主站蜘蛛池模板: 亚洲国产人成网站在线电影动漫| 亚洲国产婷婷香蕉久久久久久| 亚洲好看的理论片电影| 成人国产精品免费视频| 亚洲日韩在线观看| 一级特黄aaa大片免费看| 亚洲毛片不卡av在线播放一区| 美女裸体无遮挡免费视频网站| 日本一道在线日本一道高清不卡免费| 自拍偷区亚洲国内自拍| 国产精品va无码免费麻豆| 国产AV无码专区亚洲AV麻豆丫| 国产大片91精品免费观看男同| 羞羞视频网站免费入口| 亚洲精品无码久久不卡| 中文字幕免费观看全部电影| 亚洲国产精品福利片在线观看| 亚洲视频在线观看免费| 久久综合亚洲鲁鲁五月天| 免费不卡视频一卡二卡| 亚洲经典千人经典日产| 亚洲精品国产高清嫩草影院| 花蝴蝶免费视频在线观看高清版| 久久夜色精品国产亚洲AV动态图| 最近中文字幕国语免费完整| va天堂va亚洲va影视中文字幕| 日产乱码一卡二卡三免费| 一区二区在线免费视频| 午夜亚洲www湿好大| 成人免费毛片内射美女APP| 羞羞的视频在线免费观看| 亚洲无人区午夜福利码高清完整版| 无码中文字幕av免费放dvd| 亚洲av日韩av综合| 亚洲精品无码AV中文字幕电影网站| 久久99精品国产免费观看| 亚洲啪AV永久无码精品放毛片| 精品亚洲一区二区三区在线观看 | 国产一级高青免费| 亚洲黄色网址在线观看| 日本无卡码免费一区二区三区|