Linq Lambda表達(dá)式全面分析
在向大家詳細(xì)介紹Linq Lambda表達(dá)式之前,首先讓大家了解下expr是什么樣的東西,然后全面介紹Linq Lambda表達(dá)式。
介紹Linq Lambda表達(dá)式之前,先看一個例子:
- Expression<Func<string, bool>> expr = o => o.Length > 10;
初次接觸Linq Lambda表達(dá)式的人可能會被搞迷糊,這樣的語句到底是什么意思,怎么樣工作,原理又是什么。
逐級分析以上語句,分為兩個部分,以等號為界。
第一部分是變量類型的申明:Expression<Func<string, bool>> expr,表示expr這個變量是一個Linq Lambda表達(dá)式,這個表達(dá)式符合這樣的一種委托:bool DelegateName(string obj)。
第二部分是表達(dá)式的聲明o => o.Length > 10,這個“=>”是Lambda操作符,讀作“轉(zhuǎn)到”,必須把=>左右看成是一個整體,因?yàn)檫@實(shí)際是一個匿名方法,“=>”左邊是方法傳入?yún)?shù)的申明,右邊是函數(shù)體,如果用常規(guī)的表示方法,可以寫成如下形式:
- bool MethodName(string o)
- {
- return o.Length > 10;
- }
仔細(xì)觀察兩部分拆解以后的形式其實(shí)不難發(fā)現(xiàn),第一部分的工作是定義了一個匿名的委托,而第二部分則是符合這個匿名委托的一個方法,由于這個方法沒有明確給定名稱,因此稱為匿名方法。
那么,expr到底又是什么樣的東西。有一點(diǎn)必須明確的是,expr表示絕對不是這個匿名方法的返回值,而是這個匿名方法中所有表達(dá)式的System.Linq.Expressions.Expression形式。也就是說,在expr中,這個函數(shù)體里所有的表達(dá)式已經(jīng)被拆解成一個一個的單元,每一個單元都是一種System.Linq.Expressions.Expression的派生類。由于表達(dá)式和表達(dá)式之前存在著上下級的關(guān)系,因此所有的表達(dá)式呈現(xiàn)一種樹狀結(jié)構(gòu),稱為表達(dá)式樹。
一個匿名方法是如何轉(zhuǎn)換為表達(dá)式樹的呢?這個問題其實(shí)不用太過關(guān)心,因?yàn)镃#編譯器在對程序編譯的時候已經(jīng)將上述第二部分的內(nèi)容自動轉(zhuǎn)換為相應(yīng)的表達(dá)式樹了。上述例子中編譯的結(jié)果通過Reflector反編譯出來的內(nèi)容如下所示:
- 1. ParameterExpression CS$0$0000;
- 2. Expression<Func<string, bool>> expr = Expression.Lambda<Func<string,
bool>>(Expression.GreaterThan(Expression.Property(CS$0$0000 = Expression.
Parameter(typeof(string), "o"), (MethodInfo) methodof(string.get_Length)),
Expression.Constant(10, typeof(int))), new ParameterExpression[] { CS$0$0000 });
這串代碼看起來有點(diǎn)糊,我把代碼梳理了一下使得它更容易讀,如下所示:
- Expression<Func<string, bool>> expr;
- // 創(chuàng)建表示參數(shù)的表達(dá)式。
- ParameterExpression paramExpr = Expression.Parameter(typeof(string), "o");
- // 獲取表示System.String.Length屬性的System.Reflection.PropertyInfo對象。
- PropertyInfo propInfo = typeof(string).GetProperty
("Length", BindingFlags.Instance | BindingFlags.Public);- // 創(chuàng)建訪問System.String.Length屬性的表達(dá)式。
- MemberExpression memberExpr = Expression.Property(paramExpr, propInfo);
- // 創(chuàng)建一個表示常量10的表達(dá)式。
- ConstantExpression constExpr = Expression.Constant(10, typeof(int));
- // 創(chuàng)建表示左邊大于右邊的二分表達(dá)式。
- BinaryExpression greaterThanExpr = Expression.GreaterThan(memberExpr, constExpr);
- // 通過上述二分表達(dá)式創(chuàng)建一個Lambda表達(dá)式。
- expr = Expression.Lambda<Func<string, bool>>(greaterThanExpr, paramExpr);
是不是好麻煩啊?呵呵,好在這些工作已經(jīng)在編譯的時候完成了,不需要我們手工創(chuàng)建,除非你想動態(tài)創(chuàng)建表達(dá)式。關(guān)于如何動態(tài)創(chuàng)建表達(dá)式,我在這里就先不詳細(xì)說明了,將在下一博里再詳述。
綜上所述,對待Linq Lambda表達(dá)式,最基本一個原則是不要把表達(dá)式看成了語句的運(yùn)算結(jié)果,而應(yīng)該看成這些語句本身,也就是把語句作為對象來處理。語句和語句之間通過表達(dá)式樹來關(guān)聯(lián),而從語句轉(zhuǎn)換為表達(dá)式樹已由編譯器自動完成,不需要人工介入。
【編輯推薦】


















