匿名方法和Lambda表達式

在我們程序中,經常有這樣一些需求:

1. 需要一個臨時方法,這個方法只會使用一次,或者使用的很少。

2. 這個方法的方法體很短,以至於比方法聲明都短,寫起來實在沒勁(我將其稱之爲“一句話方法”)。

沒辦法,這樣的方法寫起來真是吃力不討好,比如一些按鈕事件處理中,有些按鈕點擊就是彈出一個對話框,或者調用一下別的什麼方法。比如下面的代碼:

this.btnRefresh.Click += new System.EventHandler(this.btnRefresh_Click);
private void btnRefresh_Click(object sender, EventArgs e)
{
    BindData();
}

這個”Refresh”按鈕就是做一下調用一下BindData()數據綁定的方法,爲此我們不得不寫一個新方法。好了,C# 2.0爲我們提供了匿名方法:
this.btnRefresh.Click += delegate(object sender, EventArgs e) { BindData(); };

沒勁的代碼沒了。想知道這種寫法的幕後麼?

其實編譯器還是在我們的後面幹了一件齷齪的事情:它爲我們產生了一個新的方法,它只是表面上爲我們節省了代碼。

private void b__0(object sender, EventArgs e)
{
    this.BindData();
}

看看這個編譯器產生的方法的名稱:

b_0,Test是這個匿名方法所放置的地方(因爲這個按鈕的時間我是放在一個Test方法裏的) 還有一點需要注意的是,如果這個匿名方法是在實例方法裏使用,那麼編譯器爲我們生成的幕後方法也是實例方法,否則就是靜態方法了。

是不是覺得匿名方法這東西很不錯,減少了很多代碼阿,但是匿名方法的使用還並不人性化,什麼是人性化呢?比如你可以用自然的語言將程序代碼讀出來,這樣纔算人性化了.在.net 2.0中System.Collections.Generic命名空間下List裏有一些新增的方法。比如Find,如果使用匿名方法我們如何調用呢:

books.Find(delegate(Book book){return book.Price < 50;});
代碼是很簡單,但是卻無法朗讀出來,來看看Lambda表達式的寫法:
books.Find(book=>book.Price<50);

這個Lambda表達式就可以這樣閱讀出來了:給你一本書,如果它的價格小於50則返回true。

好了,那我們就走進Lambda表達式吧:


將使用了Lambda表達式的程序集反編譯後,我們發現,它實際上和匿名方法沒有什麼不同。Lambda的輸入參數就對應着delegate括號裏面的參數,由於Lambda表達式可以推斷參數的類型,所以這裏的參數無需聲明。

Lambda操作符讀作”Goes to”,它後面緊跟着表達式或者是語句塊(這點和匿名方法也不同,匿名方法只能使用語句塊而不能使用表達式),下面我就用實例來說明一下有那些類型的Lambda表達式:

//x的類型省略了,編譯器可以根據上下文推斷出來,後面跟着的是表達式

x => x+1 
deleage(int x){return x+1;}
//後面跟着的是語句塊
x=>{return x+1;}
delegate(int x){return x+1;}
//輸入參數也可以帶類型,帶類型後別忘記小括號哦
(int x) => x+1
delegate(int x){return x+1;}
//也可以多個輸入參數,逗號分隔,別忘記小括號
(x,y) => x+y
delegate(int x,int y){return x+y;}
//無參的也行

() => 1

delegate(){return 1;}

對於Lambda表達式來說她的用法就是如此,但是在Lambda背後卻有很多的故事和玄機。用Lambda表達式可以構建表達式樹,而表達式樹對於Linq來說就像樹根對於樹一樣重要。在這裏就不討論表達式樹的問題了,這個東西也不是三言兩語能夠說清楚的,等待時機成熟的時候我們再來進一步討論。

Lambda表達式更多閱讀

Lambda實際上源遠流長,我們現在使用的機器都是馮-諾依曼體系的,屬於圖靈機,在那之前還有一種稱作λ演算的理論,但是圖靈機由於先被實現出來,所以大行其道,λ演算後來成就了函數式編程語言特別是Lisp,在函數式編程語言裏函數是第一等元素,函數的參數,函數的返回值都是函數,程序沒有變量,函數嵌套函數。而且函數式編程語言一直存在於象牙塔中,所以在工業界並沒有得到通用,不過近年來工業界比較喜歡“復古”風格,所以函數式編程語言也慢慢的走上了歷史的舞臺。函數式編程能解決一些命令式編程難以解決的問題(或者解決起來非常麻煩)。C#要做到函數風格編程怎麼辦?靠原來的方法定義的方式肯定是不可行的,2.0的匿名方法從某種程序上來說解決了這個問題,但還是不夠,3.0裏的Lambda終於很好的解決了,一個Lambda就是一個delegate,一個delegate指向一個方法,現在我們使用Lambda也能簡單的將方法作爲參數傳遞了,還可以層層嵌套,都是很簡單的事情了。



發佈了133 篇原創文章 · 獲贊 43 · 訪問量 144萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章