IEnumerable和IQueryable的区别以及背后的ExpressionTree表达式树
關于IEnumerable和IQueryable的區別,這事還要從泛型委托Func<T>說起。來看一個簡單的泛型委托例子:
?
class Program { static void Main(string[] args) { Func<int, bool> f = i => i > 5; Console.WriteLine(f(3)); Console.WriteLine(f(10)); Console.ReadKey(); } }?
Func<T>是"語法糖",實際上,編譯器在內部會生成一個臨時方法,再執行該方法。等同于如下:
?
class Program { static void Main(string[] args) { Func<int, bool> f = DoSth; Console.WriteLine(f(3)); Console.ReadKey(); } static bool DoSth(int i) { return i > 5; } }?
以上,.NET內部運作的路徑是:編寫C#代碼→編譯器編譯成中間語言IL→運行時JIT編譯成本地語言執行
?
■ 使用表達式樹 Expression Tree
?
可是,有時候我們希望在運行時執行代碼,該怎么辦呢?
?
.NET為我們提供了Expression Tree,允許我們在運行時執行代碼。
?
比如以上Func<int, bool> f = i => i > 5;這個表達式,Expression Tree這樣理解這個表達式:
?
○ f是Expression<Func<int, bool>>類型,級Expression<TDelegate>類型
○ =>被理解成BinaryExpression類型
○ =>左右兩邊的i被理解成ParameterExpression
○ =>右邊的5被理解成ConstantExpression
?
于是,如果我們用Expression Tree,在運行時執行代碼,可以按如下寫:
?
class Program { static void Main(string[] args) { //Func<int, bool> f = i => i > 5; ParameterExpression iParam = Expression.Parameter(typeof (int), "i"); ConstantExpression constExp = Expression.Constant(5, typeof (int)); BinaryExpression greaterThan = Expression.GreaterThan(iParam, constExp); Expression<Func<int, bool>> f = Expression.Lambda<Func<int, bool>>(greaterThan, iParam); Func<int, bool> myDele = f.Compile(); Console.WriteLine(myDele(3)); Console.WriteLine(myDele(10)); Console.ReadKey(); } }?
■ IQueryable和IEnumerable的區別???
?
現在,可以看一個IEnumerable的例子了:
?
class Program { static void Main(string[] args) { int[] intArr = new[] {1, 2, 3, 6, 8}; IEnumerable<int> result = Enumerable.Where(intArr, i => i > 5); foreach (var item in result) { Console.WriteLine(item); } Console.ReadKey(); } }?
來看一下Enumerable,實現了IEnumerable接口,它的定義:
?
再來看Queryable,實現了IQueryable接口,它的定義:
?
發現,Enumerable和Queryable很多方法同名,但參數接收的參數類型是不一樣的,Enumerable接收的參數類型是委托Func<TDelegate>,Querable接收的參數類型是Expression<Func<TDelegate>>,其類型是Expression Tree,是表達式樹。
?
所以,有關IEnumerable<T>的表達式是在編譯期確定的,有關IQueryable<T>的表達式是在運行時確定的。
?
■ 在Entity Framework應用實例中體會IQueryable<T>
?
首先在控制臺應用程序中應用Entity Framework組件。
?
創建有關Entity Framework的上下文類,類,初始數據:
?
public class Person { [Key] public int ID { get; set; } public string Name { get; set; } public int Age { get; set; } } public class MyContext : DbContext { public MyContext() : base("myConn") { Database.SetInitializer(new DbInirializer()); } public DbSet<Person> People { get; set; } } public class DbInirializer : CreateDatabaseIfNotExists<MyContext> { protected override void Seed(MyContext context) { IList<Person> people = new List<Person>(); people.Add(new Person(){Name = "張三",Age = 21}); people.Add(new Person() { Name = "李四", Age = 22 }); people.Add(new Person() { Name = "趙五", Age = 23 }); foreach (var item in people) { context.People.Add(item); } base.Seed(context); } }以上,如果轉到DbSet的定義,我們可以看到DbSet實現了IQueryable接口。??
?
配置連接字符串。
?
<configuration> <configSections> <!-- For more information on Entity Framework configuration, visit http://go.microsoft.com/fwlink/?LinkID=237468 --> <section name="entityFramework" type="System.Data.Entity.Internal.ConfigFile.EntityFrameworkSection, EntityFramework, Version=6.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089" requirePermission="false" /> </configSections> <connectionStrings> <add name="myConn" connectionString="Data Source=.;User=yourusename;Password=yourpassword;Initial Catalog=MyTest;Integrated Security=True" providerName="System.Data.SqlClient"/> </connectionStrings> <startup> <supportedRuntime version="v4.0" sku=".NETFramework,Version=v4.5" /> </startup> <entityFramework> <defaultConnectionFactory type="System.Data.Entity.Infrastructure.SqlConnectionFactory, EntityFramework" /> <providers> <provider invariantName="System.Data.SqlClient" type="System.Data.Entity.SqlServer.SqlProviderServices, EntityFramework.SqlServer" /> </providers> </entityFramework> </configuration>?
在主程序中:
?
class Program { static void Main(string[] args) { using (var context = new MyContext()) { foreach (var item in context.People) { Console.WriteLine(item.Name); } } Console.ReadKey(); } }?
現在來體會IQueryable<T>的一些特性。
?
我們知道,DbSet實現了IQuerayble接口,于是上下文的的People屬性類型是IQueryable<Person>。
?
通過,
?
IQueryable<Person> people = context.People;
?
得到的people是表達式,是sql語句,現在嘗試打印不同情況下的people表達式。
?
class Program { static void Main(string[] args) { using (var context = new MyContext()) { IQueryable<Person> people = context.People; var r = new Random(); Func<bool> rBool = () => r.Next()%2 == 0; Console.WriteLine(people); if (rBool()) { people = people.Where(p => p.Age > 21); Console.WriteLine(people); } else { people = people.OrderBy(p => p.Age); Console.WriteLine(people); } } Console.ReadKey(); } }?
由此可以看出:IQueryable呈現給我們的是表達式而不是集合,通過這個表達式可以按需加載滿足條件的數據。
總結
以上是生活随笔為你收集整理的IEnumerable和IQueryable的区别以及背后的ExpressionTree表达式树的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Oracle EM 12c
- 下一篇: 拥抱变化——从Atlas到ASP.NET