1. 程式人生 > >關於LINQ 和lambda表示式

關於LINQ 和lambda表示式

一LINQ :    既可以實現過濾資料(和lambda一樣)也可以實現查詢其他型別資料的功能 LINQ表示式的from行可以巢狀  實現表多層關聯一層層向下找的目的  (注意一對一還是一對多的對應關係 容易亂  別把上層過濾掉的內容又關聯回來了)   過濾完要ToList() 不然是IQueryable型別

   var actionList= from r in userInfo.RoleInfo     //等號前後型別不一樣  把使用者型別變成角色型別又變成許可權型別了(通過導航屬性)
         from a in r.ActionInfo
         where a.ActionTypeEnum == actionTypeEnum
         select a;                                                

二 lambda :

 動態拼接:  原文地址 http://www.cnblogs.com/green-4984/archive/2012/02/22/2362614.html

作為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());

  工具擴充套件類: 原文:http://www.cnblogs.com/Lau7/p/5451985.html

1 public static class ExpressionExt
 2 {
 3     public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> expr1, Expression<Func<T, bool>> expr2)
 4     {          
 5         return Expression.Lambda<Func<T, bool>>(Expression.AndAlso(expr1.Body, expr2.Body), expr1.Parameters);
 6     }
 7     public static Expression<Func<T, bool>> Or<T>(this Expression<Func<T, bool>> expr1,Expression<Func<T, bool>> expr2)
 8     {
 9         return Expression.Lambda<Func<T, bool>>(Expression.OrElse(expr1.Body, expr2.Body), expr1.Parameters);
10     }
11 }
複製程式碼

  上面這程式碼是我們這文章的核心,我在這裡順便解釋一下這個And擴充套件方法它是怎麼做的,它這裡先是使用Expression類中的靜態方法AndAlso把expr1和expr2的主體拼接在一起,如果AndAlso方法返回的是BinaryExpression型別的結果,而dapperLambda的條件引數需要的是Lambda表示式樹,所以這裡我們需要通過Expression.Lambda方法來構造一個委託型別來建立一個Lambda表達樹。

  那現在我們通過上面的擴充套件方法,再來優化一下我們最初舉的例子看下:

複製程式碼
1 Expression<Func<SysUser, bool>> exp1 = s => s.UserName.Contains("1") && s.Age > 0;
2 Expression<Func<SysUser, bool>> exp2 =exp1.And( s => s.IsEnable == 1);
3 using (var context = new DbContext().ConnectionString(connString))
4 {
5     var result1 = context.Select<SysUser>(exp1).QueryMany();
6     var result2 = context.Select<SysUser>(exp2).QueryMany();
7 }
複製程式碼

 對於過濾前後型別一樣的情況可以使用  僅僅起過濾作用 不改變型別 過濾完要ToList() 不然是IQueryable型別

var userMenuActionList = userActionList.Where(a => a.ActionTypeEnum == actionTypeEnum).ToList();                 //等號前後型別一樣