作為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的小節,看來有必要好好看看這個命名空間下的東東了。