閉包的由來
形成閉包有一些值得總結的非必要條件:
1、嵌套定義的函數(shù)。
2、匿名函數(shù)。
3、將函數(shù)作為參數(shù)或者返回值。
4、在.net中,可以通過匿名委托形成閉包:函數(shù)可以作為參數(shù)傳遞,也可以作為返回值返回,或者作為函數(shù)變量。而在.net中,這都可以通過委托來實現(xiàn)。這些是實現(xiàn)閉包的前提。
要說閉包的由來就不得不先說下函數(shù)式編程了。近幾年函數(shù)式編程也是比較火熱,我們先來看看函數(shù)式編程的一些基本的特性這個有助于我們理解閉包的由來。
函數(shù)式編程
函數(shù)式編程是一種編程模型,他將計算機運算看做是數(shù)學中函數(shù)的計算,并且避免了狀態(tài)以及變量的概念。這里很明顯的指出了函數(shù)式編程中最重要的就是函數(shù)而且是數(shù)學中的函數(shù),比如f(x),數(shù)學中的函數(shù)最大的特點就是只要是同樣的參數(shù)x那么我的結果必定是相等的,也就是說我們函數(shù)的返回值只是依賴于參數(shù)而不依賴于其他狀態(tài)(比如js中的全局變量就是一個干擾因素);后一句中說避免變量的概念,這句話如果從函數(shù)式編程來說不太恰當,因為這句話中的函數(shù)意思還是我們在編程語言中所使用的變量也就是一個存儲單元,而在函數(shù)式編程中變量卻是數(shù)學中變量的定義是一個值得名稱。比如,我們最基本的賦值等式 x = x+1,讓我們程序員看這是一個簡單的賦值代碼,而讓學數(shù)學的人來說這個等式是根本不成立的。 所以我們在函數(shù)式編程中是不允許多次賦值的。而這一句話也是講述了函數(shù)式編程好處的最主要的原因:
第一點、函數(shù)的結果只依賴于參數(shù)而不依賴其他狀態(tài),這樣寫的代碼很容易進行推理不容易發(fā)生錯誤,極大的方便的單元測試和調試。
第二點、因為不可變性和無狀態(tài),那么我們在處理多個線程之間就不用擔心資源的爭奪,不需要用鎖來保存狀態(tài)。
高階函數(shù)
函數(shù)式編程中函數(shù)是一等公民,和我們的口號 "萬物皆對象"有點相似,在函數(shù)式編程中,我們努力用函數(shù)來表達所有的事情,當然我們也需要函數(shù)可以傳過來傳過去這就是高階函數(shù),也就是把函數(shù)作為參數(shù)或者返回值,繼而實現(xiàn)復用,這樣即是可以把復用的粒度降到函數(shù)。c#語言中也有類似的東西--委托,當然c#中的函數(shù)跟函數(shù)式編程中的就不一樣了,但是有吸收一些函數(shù)式編程語言中的特性,比如c#中l(wèi)amda,linq。
關于函數(shù)式編程博客園有很多很好的文章介紹我就不詳說了,接下來就是引出我們今天的主題--閉包。
因為函數(shù)式編程的基礎就是lambda演算,所以這一節(jié)演算我們用lambda演算來帶出我們的主題,關于這個演算我也懂得不是太多,想要入門的同學可以看看這個 點這里
首先定義一個簡單的演算
λx.λy.x+y
如果x為1 y為2 演算過程則為
((λx.λy.x+y)1)2=(λy.1+y)2=(1+2)=3
接下來我們用到高階函數(shù)
λy . (λx . x + y)
演算過程:
((λy . (λx . x + y))1)2=((λx . x + 1))2 = (2+1)=3
可以看到這個演算中外層函數(shù)使用的是內層函數(shù),也就是說使用是一個函數(shù)作為了計算結果。ok ,我們把內層函數(shù)單獨拿出來,(λx . x + y),可以很明顯的看到如果脫離的上下文呢,我們的y是沒有值的,也就是y是沒有綁定的,也可以說y對于我們這個函數(shù)是自由的! 如果在函數(shù)式編程中,在外層函數(shù)執(zhí)行完畢之后我們的y變量就應該被銷毀,那么如果我們在內層函數(shù)中如果還需要用到y(tǒng)的話怎么辦呢? 對于這個問題,設計者則做了其他的處理:如果一個函數(shù)返回另一個函數(shù),而被返回函數(shù)又需要外層函數(shù)的變量時,不會立即釋放這個變量,而是允許被返回的函數(shù)引用這些變量。支持這種機制的語言稱為支持閉包機制,而這個內部函數(shù)連同其自由變量就形成了一個閉包(這句話是摘自其他博客,自己難得整理文字。。。)。這就是我們閉包的由來,而我們其他的語言如果有用到函數(shù)式編程的思想,并且允許函數(shù)來進行傳遞就會遇到類似的問題,所以各個語言就需要用其自己的方式來實現(xiàn)閉包!
c#中閉包的實現(xiàn)
從上一節(jié)我們也就是能總結出閉包其實就是要執(zhí)行并且包含自由變量的代碼塊(由于自由變量被包含在代碼塊中,這些自由變量以及它們引用的對象沒有被釋放)和為自由變量提供綁定的計算環(huán)境的一個結合。
然后進入我們的c#編程時刻了,我們就用簡單的例子來實現(xiàn),并且查看編譯器生成的代碼 看看c#中是怎么實現(xiàn)閉包,畢竟我們也是有委托的 。。。
首先寫一個簡單得不能在簡單的代碼
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
|
using system; namespace closure { class program { static void main( string [] args) { console.writeline(test(1)(2)); console.readkey(); } public static func< int , int > test( int x) { //作用域1 return (y) => { //作用域2 return x + y; }; } } } |
可以看到我們test的方法中傳入變量x的作用域是在1 在執(zhí)行匿名函數(shù)的時候應該是已經(jīng)釋放在作用域2就不應該存在了,而我們卻能準確的得到計算結果
說明我們的變量x確實在作用域2中還存在,接下來我們看看編譯器幫我們做了什么事情,
可以看到我們的test方法中多了一個對象 <>c__displayclass1_0 class_;
這個東西的具體定義是啥?
這個很明顯了,其實閉包只是編譯器幫我們把自由變量封裝到了一個對象中供我們作用域外使用,那我們如果去掉作用域2中使用x變量呢?
編譯器原來為了自由變量維護的對象沒了 。。。結果在意料之中。
ok,這篇文章就到此結束了,關于閉包是python中啊 js中啊 或者c#中得用處我就不細說了
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。
原文鏈接:http://www.cnblogs.com/rstar/p/closure.html