作為Delegate的更進一步的應用,Lambda讓我們的代碼更加的簡介與方便,可以方便的用Where()、Select()等擴展方法對集合進行篩選,組合。但同時也遇到了一個問題,有時候,因為用戶想要進行的條件並不是固定不便的,有時候會這么查,有時候又會組合查,同時,有時候因為數據庫設計的原因,有的字段拼接成一個很長的字符串,但是這時又要進行查詢,只要與條件有交集,那么就要提取出這條記錄,所以必須要用到動態構建Lambda表達式。
但是作為一種靜態語言,我們顯然無法用動態語法或者拼接字符串的方式來創建一個Delegate/Lambda,那么如何才能達到類似的目的呢?或許最佳的選擇就是表達式樹。
我們都知道Lambda的樣子是這樣的
i = > i < 5
在這個Lambda表達式中,i被成為Parameter,< 叫做操作符,以及一個常數 5,這樣就構建了一個Lambda表達式,那么MS專門提供了一些類來讓我們可以手工創建。
首先需要引入命名空間
using System.Linq.Expressions.Expression;
創建一個數組用來當做例子
var ints = new int []{ 1, 2 , 3 , 4 , 5 , 6 };
// 要創建形如 i => i < 5
//創建參數 i
var parameter = Expression.Parameter(typeof(int),”i”);
//創建常數 5
var constant = Expression.Constant(5);
//創建 i > 5
var bin = Expression.GreaterThan(parameter,constant);
//獲取Lambda表達式
var lambda=Expression.Lambda<Func<Int32,Boolean>>(bin,parameter);
//取得查詢結果
var query = ints.Where(lambda.Compile());
到這里就完成了一次結果的查詢,此時在程序中通過下斷點的方式來查看我們構造的表達式是否正確。
接下來創建更加復雜一些的表達式
BinaryExpression condition = null;
//要構造的表達式i==1||i==2||i==3.....
for (int i = 0; i < ints.Length; i++)
{
ConstantExpression ce = Expression.Constant(i);
if (condition == null)
{
condition = Expression.Equal(parameter, ce);
}
else
{
var right = Expression.Equal(parameter, ce);
condition = Expression.Or(condition, right);
}
}
Expression<Func<Int32, Boolean>> lambda = Expression.Lambda<Func<Int32, Boolean>>(condition, parameter);
因為現在用實體類,那么如果是實體類進行構造的話需要這樣來做
//p => p.Name == "1" && p.Address == "2"
ParameterExpression parameter1 = Expression.Parameter(typeof(Person), "p");
ConstantExpression constant1 = Expression.Constant("1");
ConstantExpression constant2 = Expression.Constant("1");
MemberExpression member = Expression.PropertyOrField(parameter1, "Name");
var query1 = Expression.Equal(member, constant1);
var query2 = Expression.Equal(Expression.PropertyOrField(parameter1, "Address"), constant2);
var query = Expression.And(query1, query2);
var lambda1 = Expression.Lambda<Func<Person, Boolean>>(query, parameter1);
var list = MethodExtend.GetUser(lambda1.Compile());
以上是對動態創建Lambda的小節,看來有必要好好看看這個命名空間下的東東了。