转载

20.C#LINQ基础和简单使用(十一章11.1-11.2)

终于看到了第11章,之前虽然也有看过,但没有太仔细,在工作中也偶尔会使用,但不明白其中的原理,那现在就来讲讲LINQ,做一做书虫~~

首先先了解下LINQ的三个要点:

  1. LINQ不能把非常复杂的查询表达式转换成一行代码
  2. 使用LINQ不意味着你从此不再需要使用SQL
  3. LINQ不可能魔法般地让你成为架构天才

序列是LINQ的基础,在你看到一个查询表达式的时候,应该要想到它所涉及的序列:一开始总是存在至少一个序列,且通常在中间过程会转换成其他序列,也可能和其他序列连接在一起。

 1 class Car  2 {  3     public string Owner { get; set; }  4     public double Mileage { get; set; }  5 }  6   7 class Program  8 {  9     static void Main(string[] args) 10     { 11         var list = new List<Car> { 12             new Car() { Owner="a",Mileage=100}, 13             new Car() { Owner="b",Mileage=200}, 14             new Car() { Owner="c",Mileage=300}, 15             new Car() { Owner="d",Mileage=400}, 16             new Car() { Owner="e",Mileage=500}, 17             new Car() { Owner="f",Mileage=600} 18         }; 19  20         var result = from e in list 21                      where e.Mileage > 200 22                      select e; 23  24         result.ToList().ForEach(x => Console.WriteLine(x.Owner)); 25  26         Console.ReadKey(); 27  28     } 29 }

上述的表达式作用: 从一个汽车序列中找到里程大于200的车 。下一句代码使用Lambda表达式来打印筛选出来的车主名。

  • 延迟执行和流处理

上述中的查询表达式被创建的时候,不会立刻处理数据,也不会访问原始的汽车序列,而是在内存中生成这个查询的表现形式,这个特点叫延迟执行 。如下,在result.ToList()执行时,才会开始对序列的操作。

  • 投影

select expression 格式的语句,称为投影。

1 result.Where(e => e.Mileage > 200).Select(e => e);

查询表达式会编译器转译为上面的代码,可以看到它们是由扩展方法和Lambda表达式组成。在任何时候, 参数(大多情况)都是委托类型,编译器将用Lambda表达式作为实参,并尽量找到具有合适签名的方法 。再来看下我们的表达式

1 var result = from e in list 2              where e.Mileage > 200 3              select e;

from、where、in、select为查询表达式上下文关键字,e为范围变量、list为数据源序列、select e为返回投影。再看一个返回投影的例子。

1 var result1 = from e in list 2               where e.Mileage > 200 3               select e.Owner;

这里的select e.Owner,即返回每辆车主的名字,那么result1的类型为IEnumerable<string>

  • Cast方法和OfType方法
 1 class Bigtruck :Car  2 {  3     public double Volume { get; set; }  4   5 }  6   7 var list0 = new List<Car> {  8     new Car() { Owner="a",Mileage=100},  9     new Bigtruck() { Owner="b",Mileage=200,Volume=100.1}, 10     new Bigtruck() { Owner="c",Mileage=300,Volume=100.2}, 11     new Bigtruck() { Owner="d",Mileage=400,Volume=100.3}, 12     new Bigtruck() { Owner="e",Mileage=500,Volume=100.4}, 13     new Bigtruck() { Owner="f",Mileage=600,Volume=100.5} 14 }; 15  16 var castList = list0.Cast<Car>(); 17 var oftypeList = list0.OfType<Bigtruck>(); 18  19 Console.WriteLine(castList.Count());  //6 20 Console.WriteLine(oftypeList.Count());  //5

使用Cast<T>()会把列表中元素转换成T类型,遇到不能转换的元素将报错,而OfType<T>()会尝试转换每个元素到T类型,遇到不能转换的元素则跳过 。当显式声明使用范围变量时,会在转译时调用Cast()方法。如

1 var result2 = from Car e in list 2               select e; 3 result2 = list.Cast<Car>().Select(e => e);

表达式会转译为第三行的代码。

  • 概念
  1. LINQ是以数据列表为基础,在任何地方可能的地方进行流处理
  2. 创建一个查询表达式,不会立刻执行,大部分操作都会延迟执行
  3. C#3的查询表达式包括一个把表达式转换成普通C#代码的预处理阶段,接着使用类型推断、重载、Lambda表达式等这些常规操作来恰当地对转换后的代码进行编译
  4. 在查询表达式中声明的变量:它们仅仅是范围变量,通过它们你可以在查询表达式内部一致地引用数据

请斧正。

正文到此结束
Loading...