首页 > 代码库 > 20.C#LINQ基础和简单使用(十一章11.1-11.2)

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 list21                      where e.Mileage > 20022                      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 list2              where e.Mileage > 2003              select e;

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

1 var result1 = from e in list2               where e.Mileage > 2003               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());  //620 Console.WriteLine(oftypeList.Count());  //5

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

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

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

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

  请斧正。

 

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