在我們程式中,經常有這樣一些需求:
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(); };沒勁的程式碼沒了。想知道這種寫法的幕後黑手麼?其實編譯器還是在我們的後面幹了一件齷齪的事情:它為我們產生了一個新的方法,它只是表面上為我們節省了程式碼。privatevoidb__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 => x+1deleage(int x){return x+1;}//後面跟著的是語句塊x=>{return x+1;}delegate(int x){return x+1;}//輸入引數也可以帶型別,帶型別後別忘記小括號哦(int x) => x+1delegate(int x){return x+1;}//也可以多個輸入引數,逗號分隔,別忘記小括號(x,y) => x+ydelegate(int x,int y){return x+y;}//無參的也行() => 1delegate(){return 1;}對於Lambda表示式來說她的用法就是如此,但是在Lambda背後卻有很多的故事和玄機。用Lambda表示式可以構建表示式樹,而表示式樹對於Linq來說就像樹根對於樹一樣重要。在這裡就不討論表示式樹的問題了,這個東西也不是三言兩語能夠說清楚的,等待時機成熟的時候我們再來進一步討論。
在我們程式中,經常有這樣一些需求:
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(); };沒勁的程式碼沒了。想知道這種寫法的幕後黑手麼?其實編譯器還是在我們的後面幹了一件齷齪的事情:它為我們產生了一個新的方法,它只是表面上為我們節省了程式碼。privatevoidb__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 => x+1deleage(int x){return x+1;}//後面跟著的是語句塊x=>{return x+1;}delegate(int x){return x+1;}//輸入引數也可以帶型別,帶型別後別忘記小括號哦(int x) => x+1delegate(int x){return x+1;}//也可以多個輸入引數,逗號分隔,別忘記小括號(x,y) => x+ydelegate(int x,int y){return x+y;}//無參的也行() => 1delegate(){return 1;}對於Lambda表示式來說她的用法就是如此,但是在Lambda背後卻有很多的故事和玄機。用Lambda表示式可以構建表示式樹,而表示式樹對於Linq來說就像樹根對於樹一樣重要。在這裡就不討論表示式樹的問題了,這個東西也不是三言兩語能夠說清楚的,等待時機成熟的時候我們再來進一步討論。