日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

IEnumerable和IQueryable的区别以及背后的ExpressionTree表达式树

發布時間:2023/12/20 编程问答 28 豆豆
生活随笔 收集整理的這篇文章主要介紹了 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表达式树的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。