LINQ中延时执行的Enumerable类方法成员

开发 后端
LINQ标准查询运算法是依靠一组扩展方法来实现的。而这些扩展方法分别在System.Linq.Enumerable和System.Linq.Queryable这连个静态类中定义。

Enumerable的扩展方法采用线性流程,每个运算法会被线性执行。这种执行方法如果操作类似关系型数据库数据源,效率会非常低下,所以Queryable重新定义这些扩展方法,把LINQ表达式拆解为表达式树,提供程序就可以根据表达式树生成关系型数据库的查询语句,即SQL命令,然后进行相关操作。

  每个查询运算符的执行行为不同,大致分为立即执行和延时执行。延时执行的运算符将在枚举元素的时候被执行。

  Enumerable类位于程序集System.Core.dll中,System.Linq命名空间下,并且直接集成自System.Object,存在于3.5及以上的.NET框架中。Enumerable是静态类,不能实例化和被继承,其成员只有一组静态和扩展方法。

  LINQ不仅能够查询实现IEnumerable<T>或IQueryable<T>的类型,也能查询实现IEnumerable接口的类型。关于Enumerable方法的详细说明,请参考MSDN  Enumerable 类

  理解LINQ首先必须理解扩展方法

  msdn是这样规定扩展方法的:“扩展方法被定义为静态方法,但它们是通过实例方法语法进行调用的。 它们的***个参数指定该方法作用于哪个类型,并且该参数以 this 修饰符为前缀。”

下面给个扩展方法的例子如下:

  1. using System; 
  2. using System.Collections.Generic; 
  3. using System.Linq; 
  4. using System.Text; 
  5.  
  6. namespace 扩展方法 
  7.     /// <summary> 
  8.     /// 为string类型定义一个扩展方法 
  9.     /// </summary> 
  10.     static class Helper 
  11.     {  
  12.         public static string MyExtenMethod(this string s) 
  13.         { 
  14.             return s.Substring(0, 2); 
  15.         } 
  16.     } 
  17.     class Program 
  18.     { 
  19.         static void Main(string[] args) 
  20.         { 
  21.             string s = "扩展方法示例"
  22.             Console.WriteLine(s.MyExtenMethod());//调用 
  23.             Console.ReadKey(false); 
  24.         } 
  25.     } 

程序的运行结果如下:

 -----插曲,想到了就加进来,有助于理解开头的几段话及LINQ原理

 ---------------------------------------------------------------”

 为了方便理解和记忆,DebugLZQ将常用的延时执行的Enumerable类方法成员分了下组,具体如下:

1.Take用于从一个序列的开头返回指定数量的元素

2.TakeWhile 用于获取指定序列从头开始符合条件的元素,直到遇到不符合条件的元素为止

3.Skip跳过序列中指定数量的元素

4.SkipWhile 用于跳过序列总满足条件的元素,然会返回剩下的元素

  1. using System; 
  2. using System.Collections.Generic; 
  3. using System.Linq; 
  4. using System.Text; 
  5.  
  6. namespace 延时执行的Enumerable类方法 
  7.     /// <summary> 
  8.     /// 延时执行的Enumerable类方法 
  9.     /// DebugLZQ 
  10.     /// http://www.cnblogs.com/DebugLZQ 
  11.     /// </summary> 
  12.     class Program 
  13.     { 
  14.         static void Main(string[] args) 
  15.         { 
  16.             string[] names = { "DebugLZQ","DebugMan","Sarah","Jerry","Tom","Linda","M&M","Jeffery"}; 
  17.             //1.Take用于从一个序列的开头返回指定数量的元素 
  18.             // 
  19.             //a.在数组上直接使用Take方法 
  20.             foreach (string name in names.Take(3)) 
  21.             { 
  22.                 Console.Write("{0}    ", name);  
  23.             } 
  24.             Console.WriteLine(); 
  25.             Console.WriteLine("-----"); 
  26.             //b.在LINQ返回的IEnumerable<T>序列上使用Take方法 
  27.             var query = from string name in names 
  28.                         where name.Length <=3 
  29.                         select name; 
  30.             foreach (string  name in query.Take(1)) 
  31.             { 
  32.                 Console.Write("{0}    ",name); 
  33.             } 
  34.             Console.WriteLine(); 
  35.             Console.WriteLine("----------------------------"); 
  36.             Console.ReadKey(false); 
  37.             //2.TakeWhile 用于获取指定序列从头开始符合条件的元素,直到遇到不符合条件的元素为止 
  38.             // 
  39.             var takenames = names.TakeWhile(n => n.Length>4); 
  40.             var takenames2 = names.TakeWhile((n,i)=>n.Length<10&&i<3); 
  41.             foreach (string name in takenames) 
  42.             { 
  43.                 Console.Write("{0}    ", name); 
  44.             } 
  45.             Console.WriteLine(); 
  46.             Console.WriteLine("-----"); 
  47.             foreach (string name in takenames2) 
  48.             { 
  49.                 Console.Write("{0}    ", name); 
  50.             } 
  51.             Console.WriteLine(); 
  52.             Console.WriteLine("----------------------------"); 
  53.             Console.ReadKey(false); 
  54.             //3.Skip跳过序列中指定数量的元素 
  55.             // 
  56.             foreach (string name in names.Skip(5)) 
  57.             { 
  58.                 Console.Write("{0}    ", name); 
  59.             } 
  60.             Console.WriteLine(); 
  61.             Console.WriteLine("-----"); 
  62.             var query_skip = (from name in names 
  63.                               where name.Length >= 3 
  64.                               select name).Skip(2); 
  65.             foreach (string name in query_skip.Skip(2) ) 
  66.             { 
  67.                 Console.Write("{0}    ", name); 
  68.             } 
  69.             Console.WriteLine(); 
  70.             Console.WriteLine("----------------------------"); 
  71.             Console.ReadKey(false); 
  72.             //4.SkipWhile 用于跳过序列总满足条件的元素,然会返回剩下的元素 
  73.             //跳过名字长度大于3的 
  74.             var takenames_SkipWhile = names.SkipWhile(n => n.Length >3); 
  75.             foreach (string name in takenames_SkipWhile) 
  76.             { 
  77.                 Console.Write("{0}    ", name); 
  78.             } 
  79.             Console.WriteLine(); 
  80.             Console.WriteLine("-----"); 
  81.             var takenames_SkipWhile2 = names.SkipWhile((n,i)=>n.Length>3&&i>2); 
  82.             foreach (string name in takenames_SkipWhile2) 
  83.             { 
  84.                 Console.Write("{0}    ", name); 
  85.             } 
  86.             Console.WriteLine(); 
  87.             Console.WriteLine("----------------------------"); 
  88.             Console.ReadKey(false); 
  89.  
  90.             //小结Take、Skip获得第N到第M个元素 
  91.             var names_TakeAndSkip = names.Skip(5).Take(3); 
  92.  
  93.             var names_TakeAndSkip2 = (from name in names 
  94.                                       select name).Skip(5).Take(3); 
  95.  
  96.             foreach (string name in names_TakeAndSkip) 
  97.             { 
  98.                 Console.Write("{0}    ", name); 
  99.             } 
  100.             Console.WriteLine(); 
  101.             Console.WriteLine("-----"); 
  102.             foreach (string name in names_TakeAndSkip2) 
  103.             { 
  104.                 Console.Write("{0}    ", name); 
  105.             } 
  106.             Console.WriteLine(); 
  107.             Console.WriteLine("----------------------------"); 
  108.             Console.ReadKey(false); 
  109.  
  110.         } 
  111.     } 

程序中有详细的注释不再多做说明,程序运行结果如下:

5.Reverse用于翻转序列中的元素的顺序

6.Distinct过滤掉重复的元素

  1. using System; 
  2. using System.Collections.Generic; 
  3. using System.Linq; 
  4. using System.Text; 
  5.  
  6. namespace Reverse_Distinct等 
  7.     /// <summary> 
  8.     /// DebugLZQ 
  9.     /// http://www.cnblogs.com/DebugLZQ 
  10.     /// </summary> 
  11.     class Program 
  12.     { 
  13.         static void Main(string[] args) 
  14.         { 
  15.             string[] names = { "DebugLZQ""Jerry""Sarah""Jerry""Tom""Linda""M&M""Jeffery" }; 
  16.             //5.Reverse用于翻转序列中的元素的顺序 
  17.            string str = "反转字符串"
  18.  
  19.            var strre = str.ToCharArray().Reverse(); 
  20.            var takenames = names.Reverse(); 
  21.  
  22.            foreach (var c in strre) 
  23.            { 
  24.                Console.Write(c); 
  25.            } 
  26.            Console.WriteLine(); 
  27.            Console.WriteLine("-----"); 
  28.            foreach (var c in takenames ) 
  29.            { 
  30.                Console.WriteLine(c); 
  31.            } 
  32.            Console.WriteLine("----------------------------"); 
  33.            Console.ReadKey(false); 
  34.  
  35.             //6.Distinct  过滤掉重复的元素 
  36.            var takenames_Distinct = names.Distinct(); 
  37.  
  38.            foreach (var c in takenames_Distinct) 
  39.            { 
  40.                Console.WriteLine(c); 
  41.            } 
  42.            Console.WriteLine("----------------------------"); 
  43.            Console.ReadKey(false); 
  44.         } 
  45.     } 

程序的运行结果如下:

7.Union用于合并两个序列,并去掉重复项

8.Concat用于连接两个序列,不会去掉重复项

9.Intersect用于获得连个序列的交集

10.Except用于获得两个结合的差集

  1. using System; 
  2. using System.Collections.Generic; 
  3. using System.Linq; 
  4. using System.Text; 
  5.  
  6. namespace Union_Concat_Intersect_Except 
  7.     /// <summary> 
  8.     /// DebugLZQ 
  9.     /// http://www.cnblogs.com/DebugLZQ 
  10.     /// </summary> 
  11.     class Program 
  12.     { 
  13.         static void Main(string[] args) 
  14.         { 
  15.             string[] names1 = { "DebugLZQ""Jerry""Sarah""Jerry""Tom""Linda""M&M""Jeffery" }; 
  16.             string[] names2 = { "DebugLZQ""Jerry""Sarah" }; 
  17.  
  18.             //7.Union用于合并两个序列,并去掉重复项 
  19.             var names_Union = names1.Union(names2); 
  20.  
  21.             //8.Concat用于连接两个序列,不会去掉重复项 
  22.             var names_Concat = names1.Concat(names2); 
  23.  
  24.             //9.Intersect用于获得连个序列的交集 
  25.             var names_Intersect = names1.Intersect(names2); 
  26.  
  27.             //10.Except用于获得两个结合的差集 
  28.             var names_Except = names1.Except(names2); 
  29.  
  30.             foreach (string name in names_Union) 
  31.             { 
  32.                 Console.WriteLine(name); 
  33.             } 
  34.             Console.WriteLine("-----"); 
  35.             Console.ReadKey(false); 
  36.             foreach (string name in names_Concat) 
  37.             { 
  38.                 Console.WriteLine(name); 
  39.             } 
  40.             Console.WriteLine("-----"); 
  41.             Console.ReadKey(false); 
  42.             foreach (string name in names_Intersect) 
  43.             { 
  44.                 Console.WriteLine(name); 
  45.             } 
  46.             Console.WriteLine("-----"); 
  47.             Console.ReadKey(false); 
  48.             foreach (string name in names_Except) 
  49.             { 
  50.                 Console.WriteLine(name); 
  51.             } 
  52.             Console.WriteLine("-----"); 
  53.             Console.ReadKey(false); 
  54.         } 
  55.     } 

程序的运行结果如下:

11.Range 用于生成指定范围内的“整数”序列

12.Repeat用于生成指定数量的重复元素

13.Empty 用于获得一个指定类型的空序列

14.DefaultIfEmpty 用于获得序列,如果为空,则添加一个默认类型元素

  1. using System; 
  2. using System.Collections.Generic; 
  3. using System.Linq; 
  4. using System.Text; 
  5.  
  6. namespace Range_Empty_DefalultIfEmpty 
  7.     /// <summary> 
  8.     /// DebugLZQ 
  9.     /// http://www.cnblogs.com/DebugLZQ 
  10.     /// </summary> 
  11.     class Program 
  12.     { 
  13.         static void Main(string[] args) 
  14.         { 
  15.             //11.Range 用于生成指定范围内的“整数”序列 
  16.             var num2 = Enumerable.Range(10, 15); 
  17.  
  18.             //12.Repeat用于生成指定数量的重复元素 
  19.             var guest = new {Name="橙子",Age=25 }; 
  20.             var Guests = Enumerable.Repeat(guest, 5); 
  21.  
  22.             //13.Empty 用于获得一个指定类型的空序列 
  23.             var empty = Enumerable.Empty<string>(); 
  24.  
  25.             //14.DefaultIfEmpty 用于获得序列,如果为空,则添加一个默认类型元素 
  26.             //a 
  27.             var intempty = Enumerable.Empty<int>(); 
  28.             Console.WriteLine(intempty.Count()); 
  29.             Console.WriteLine("-----------"); 
  30.             foreach (var n in intempty) 
  31.             { 
  32.                 Console.WriteLine(n);             
  33.             } 
  34.             Console.WriteLine("-----------"); 
  35.             Console.WriteLine(intempty.DefaultIfEmpty().Count()); 
  36.             Console.WriteLine("-----------"); 
  37.             foreach (var n in intempty.DefaultIfEmpty()) 
  38.             { 
  39.                 Console.WriteLine(n); 
  40.             } 
  41.             Console.WriteLine("--------------------------"); 
  42.             Console.ReadKey(false); 
  43.             //b 
  44.             string[] names = { "DebugLZQ""DebugMan""Sarah""Jerry""Tom""Linda""M&M""Jeffery" }; 
  45.             var query = from name in names 
  46.                         where name == "LBJ" 
  47.                         select name; 
  48.             Console.WriteLine(query.Count()); 
  49.             Console.WriteLine(query.DefaultIfEmpty().Count());//默认为null 
  50.             foreach (var n in query.DefaultIfEmpty()) 
  51.             { 
  52.                 Console.WriteLine(n); 
  53.             } 
  54.             Console.WriteLine("---------------"); 
  55.             Console.ReadKey(false); 
  56.             //c指定一个默认值 
  57.             foreach (var n in intempty.DefaultIfEmpty(100)) 
  58.             { 
  59.                 Console.WriteLine(n); 
  60.             } 
  61.             Console.WriteLine("--------------------------"); 
  62.             Console.ReadKey(false); 
  63.  
  64.             foreach (var n in query.DefaultIfEmpty("James")) 
  65.             { 
  66.                 Console.WriteLine(n); 
  67.             } 
  68.             Console.ReadKey(false); 
  69.         } 
  70.     } 

程序的运行结果如下:

15.OfType筛选指定类型的元素

16.Cast类型转换

17.AsEnumerable有些数据源类型不支持Enumerable的部分查询关键字,需要转换下,譬如IQueryable

  1. using System; 
  2. using System.Collections.Generic; 
  3. using System.Linq; 
  4. using System.Text; 
  5. using System.Collections; 
  6.  
  7. namespace Cast_OfType_AsEnumerable 
  8.     /// <summary> 
  9.     /// DebugLZQ 
  10.     /// http://www.cnblogs.com/DebugLZQ 
  11.     /// </summary> 
  12.     class Program 
  13.     { 
  14.         static void Main(string[] args) 
  15.         { 
  16.             ArrayList names = new ArrayList(); 
  17.             names.Add("DebugLZQ"); 
  18.             names.Add("Jerry"); 
  19.             names.Add(100); 
  20.             names.Add(new {Name="LZQ",Age=26}); 
  21.             names.Add(new Stack()); 
  22.             //15.OfType筛选指定类型的元素 
  23.             var takenames = names.OfType<string>(); 
  24.  
  25.             //16.Cast类型转换 
  26.             var takenames2 = names.OfType<string>().Cast<string>(); 
  27.  
  28.             //17.AsEnumerable 
  29.             var takenames3 = takenames2.AsEnumerable(); 
  30.  
  31.             foreach (var name in takenames3) 
  32.             { 
  33.                 Console.Write("{0}  ",name); 
  34.             } 
  35.             Console.ReadKey(false); 
  36.  
  37.         } 
  38.     } 

程序运行结果如下:

 延时执行,顾名思义就是不是立即执行,即不是在查询语句定义的时候执行,而是在处理结果集(如遍历)的时候执行,在Enumerable类方法成员中,除了本节总结的这常用的17个外,前面博文---LINQ基本子句 中总结的8个基本子句也都是延时执行的。注意延时执行的查询程序的执行流程。

原文链接:http://www.cnblogs.com/DebugLZQ/archive/2012/11/08/2759543.html

【编辑推荐】

  1. Linq to xml操作XML
  2. XML之父解读未来互联网"游戏化"的三个真谛
  3. Ajax和Web服务数据格式:XML SOAP HTML
  4. 超强解析XML——简单直接的来
  5. 解析PHP中的XML数据

 

责任编辑:彭凡 来源: 博客园
相关推荐

2009-09-16 17:41:56

Linq延时执行

2024-04-02 11:34:09

成员对象封闭类C++

2024-02-22 18:07:17

C++静态成员代码

2009-09-11 15:12:26

LINQ执行存储过程

2009-09-15 17:16:58

LINQ查询操作符

2009-08-13 17:06:37

C#扩展方法Enumerable.

2009-08-18 17:39:16

Enumerable.

2010-02-01 17:31:06

C++类成员

2009-08-27 13:56:03

IEnumerable

2009-09-16 10:27:35

LINQ Custom

2024-06-03 08:26:34

Android开发监听器

2009-09-14 10:35:15

Linq内部执行原理

2012-09-18 09:39:57

Linq项目高效

2009-09-09 13:18:26

Linq Submit

2009-09-14 16:41:23

LINQ To XML

2009-09-09 09:59:08

Linq调用LoadP

2009-09-15 13:20:34

LINQ DataCo

2010-06-03 08:55:43

LINQ

2009-09-17 08:47:00

Linq插入数据

2009-09-17 11:32:52

LINQ调用存储过程
点赞
收藏

51CTO技术栈公众号