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