LINQ入门(中篇)

来自森大科技官方博客
http://www.cnsendblog.com/index.php/?p=163
GPS平台、网站建设、软件开发、系统运维,找森大网络科技!
http://cnsendnet.taobao.com

到现在为止你还未触碰LINQ,那进来吧 —— LINQ入门(中篇)

  在上篇中简单的分享了LINQ的基础概念及基础语法,如果没有阅读过上篇的朋友可以点击这里。感谢大家的支持,本篇我们将更进一步的学习LINQ的一些相关特性及应用方法。废话不多说,请往下阅读吧。

延迟加载

  在上篇中简单的和大家提到了LINQ具有一个很有意思的特性那就是“延迟加载”(或“延迟计算”),什么是延迟加载呢?先看来自官方的描述:延迟执行意味着表达式的计算延迟,直到真正需要它的实现值为止。是不是觉得有点生涩难理解呢?按照我个人的理解通俗的讲就是,每当我们编写好一段LINQ表达式时,此时这个表达式所代表的序列变量仅仅只是一个代理,编译器在执行编译时根本就不鸟这段代码,检查完语法正确性后直接跳过,直到代码在编译器动态运行序列变量在其他代码块被调用时,它所代理的linq表达式才会执行。啊~~看到这里你是不是要晕了,到底要怎么理解啊,无废话上代码:

1 // 已知一个序列
 2 var array = new int[] {1, 2, 3};
 3
 4 // 编写一段LINQ表达式获得一个序列变量query
 5 // 注意,这个变量仅仅是一个代理,在执行编译的时候,编译器检查完
 6 // 该代码的正确性后直接跳过,不再理会
 7 var query = from arr in array
 8             where arr > 1
 9             select arr;
10
11 // 调用上述序列变量query,此时上述的LINQ表达才会执行。注意此时已是在
12 // 编译器Runtime 的情况下执行
13 foreach(var q in query)
14     Console.WriteLine(q.ToString());

如果你觉得上述例子不能让你有个深刻的理解,那么请看来自MSDN的例子

1 public static class LocalExtensions
 2 {
 3     public static IEnumerable<string>
 4       ConvertCollectionToUpperCase(this IEnumerable<string> source)
 5     {
 6         foreach (string str in source)
 7         {
 8             Console.WriteLine("ToUpper: source {0}", str);
 9             yield return str.ToUpper();
10         }
11     }
12 }
13
14 class Program
15 {
16     static void Main(string[] args)
17     {
18         string[] stringArray = { "abc", "def", "ghi" };
19         // 这里方法 ConvertCollectionToUpperCase 是不会在编译时进行调用核查的,直到下面的foreach调用变量 q 此方法才会执行
20         var q = from str in stringArray.ConvertCollectionToUpperCase()
21                 select str;
22
23         foreach (string str in q)
24             Console.WriteLine("Main: str {0}", str);
25     }
   }

注意,ConvertCollectionToUpperCase 是一个静态扩展方法,后续讲解,如果你对.net 2.0 的 yeild 不熟悉的网上查阅吧,这里就不做介绍了。

// 输出结果

// ToUpper: source abc

// Main: str ABC

// ToUpper: source def

// Main: str DEF

// ToUpper: source ghi

// Main: str GHI

  小结,延迟加载有好也有坏,由于是在Runtime的情况下执行序列,所以就容易造成未知异常,断点打错等等,所以编码LINQ是一定要考虑到它的这个特性。

lambda 表达式

  了解完延迟加载后,那么现在我们需要简单的学习一下.net 3.5 给我们带来的新特性lambda表达式,在上篇的评论中,有园友问lambda和linq有什么关系,在这里其实他们没有任何关系,是完全不同的东西,但是我们为什么要掌握它呢?因为在后续的学习中会使用大量的lambda表达,他可以使我们的代码更优雅更有可读性,大大提高了我们的编码效率。

  那么在学习lambda之前,先来回顾一下.net 2.0给我们带来的委托 delegate ,这个你一定不会感到陌生吧,而且一定会常用他。对于委托这里就不做详细的介绍了,要复习委托的在网上查阅吧。通过委托,我们可以得到一个东西“匿名方法”。咦,是不是觉得很眼熟,呵呵,用代码来加深回忆吧

public delegate void SomeDelegate1;
public delegate void SomeDelegate2(arg 1, arg 2);

// 匿名方法
SomeDelegate1 del1 += delegate() {...};
SomeDelegate2 del2 += delegate(arg1, arg2) {...}

上面的代码中我们看到在.net 2.0时代,我们可以通过delegate创建匿名方法,提高编码的灵活性,那么lambda和这个有什么关系呢,lambda对匿名方法进行了升华。看代码:

public delegate void SomeDelegate1;
public delegate void SomeDelegate2(arg 1, arg 2);

// 匿名方法
SomeDelegate1 del1 += () => {...};
SomeDelegate2 del2 += (arg1, arg2) => {...}

呵呵,是不是觉得有点不可思议呢,言归正传什么是lambda表达式呢,来自官方的定义:“Lambda 表达式是一个匿名函数,它可以包含表达式和语句,并且可用于创建委托或表达式树类型。所有 Lambda 表达式都使用 Lambda 运算符=>,该运算符读为“goes to” Lambda 运算符的左边是输入参数(如果有),右边包含表达式或语句块。 Lambda 表达式 x => x * x 读作“x goes to x times x”。在定义里提到了表达式树,这是高阶晋级的话题,这里就不做讨论了,我们先把精力放在入门与实战应用上。

常规的lambda表达式如下:

(parameters) => {expression}

当指定的委托类型没有参数是表达式可以如下

() => {expression} 例:() => {/*执行某些方法*/}

如果表达右侧花括号里只有一个表达例如一元表达式,二元表达式等等,又或者是一个方法时那么花括号可以省略如下:

(x) => x; // 最简表达式

(x, y) => x == y;

() => SomeMethod();

注意,如果右侧的表达式存在花括号"{}",而且委托是具有返回类型的,那么表达式必须带上 return 关键字,如下:

(x, y) => {return x == y;};

到此我们已对lambad 表达式有了一定的掌握与了解。那么我们扩展一下,在.net 3.5中,ms 给我们提供了两个泛型委托 分别是 Fun<T> 和 Action <T> 他们可以帮助我们省去了返回创建常用委托的麻烦,提高编码效率。

共同点:它们至多提供委托传递6个参数(任意类型);

不同点:Fun 要求必须具有返回类型,而Action则必须不返回类型,规定返回 void

示例:

Fun<int, int, bool> fun = (a, b) => a==b;

Action<string> action = (p) => Console.Write(p);

小结,lambda 对我个人而言是个又爱又恨啊,不过爱多一点,它使我们写更少的代码做更多的事,但是在调试时一旦修改表达式内容,那么当前调试要么停止,要么重新开始,ms要是在这方面做得更完美些就好啦。不过它也间接提醒我们要有好的编码设计思维。

静态扩展方法

  说完lambda,那么我们就进一步了解一下.net 3.5的另一个新特性“静态扩展方法”,什么是静态扩展方法呢,官方定义:扩展方法使您能够向现有类型“添加”方法,而无需创建新的派生类型、重新编译或以其他方式修改原始类型。 扩展方法是一种特殊的静态方法,但可以像扩展类型上的实例方法一样进行调用。简单的说就是我们可以向一个已知的类型在不通过继承,复写等操作的情况下添加一个方法,以便类型的实例可以直接使用该方法。示例如下:

static class A
{
    // 这里的p1,p2 仅作为示例,实际中我们不一定需要
    public static int ExtendMethod1(this string input,string p1, string p2)
    {
         return int.Parse(input + p1 + p2);
    }

     // 泛型方法
     public static TOutput, ExtendMethod2<TOutput>(this object obj);
     {
           return (TOutput)obj;
     }
}

class B
{
  void Main()
   {
      var a = "1";
      var result = a.ExtendMethod1("2","3");
      // result:123
   }
}

注意,方法的 static 是必须的,而且需在静态类里。第一个参数 this 是必须的,紧跟着 this 后面的需要扩展的类型实例参数,也是必须的。至于后面的方法调用传递参数就因个人所需了。

  既然我们学习了静态扩展方法,那么它和LINQ又有什么关系呢?在System.Linq的命名空间中提供了大量的静态扩展方法,这些静态方法本身就对linq表达式的封装,这样我们就可以省去了编写简单的linq表达式的步骤。如下:

var array = new int[]{1,2,3,4,5};

var query1 = from arr in array

select arr;

var query2 = array.Select(e => e);

上面的示例中 query1 和 query2 是等价的,通过 query2 我们是不是又可以偷懒了很多,呵呵。

再来点带where的

var array = new int[]{1,2,3,4,5};

var query1 = from arr in array

where arr > 2

select arr;

var query2 = array.Where(e => e > 2);

再来一个复合型的

var array = new int[]{1,2,3,4,5};

var max = (from arr in array.Where(e => e > 2)

select arr).Max();

是不是觉得很cool。由于篇幅的关系在这里就不逐一的去接受这些静态方法了,下面是一些常用的静态方法列表,感兴趣的去MSDN查阅详细吧。

Aggregate , All , Any , AsEnumerable , Average , Cast , Concat , Contains, Count, DefaultIfEmpty , Distinct , ElementAt, ElementAtOrDefault ,Empty , Except , First, FirstOrDefault , GroupBy , GroupJoin , Intersect , Join , Last , LastOrDefault , LongCount , Max , Min , OfType ,OrderBy ,OrderByDescending , Range , Repeat , Reverse , Select , SelectMany , SequenceEqual , Single , SingleOrDefault , Skip , SkipWhile , Sum ,Take, TakeWhile , ThenBy , ThenByDescending , ToArray , ToDictionary , ToList, ToLookup, Union,Where

Cast<T> OfType<T> 静态扩展方法

  在最后我们还是要注意两个常用的静态方法Cast<T>, OfType<T> 。它们的共同点是都能把非IEnumerable<T> 类型的集合转换成IEnumerable<T>类型,然后再

进行LINQ操作,如下

var dt = new DataTable();
dt.Columsn.Add("A", typeof(int));

var newRow1 = dt.NewRow();
newRow1["A"] = 1;

var newRow2 = dt.NewRow();
newRow2["A"] = 2;

dt.Rows.Add(newRow1);
dt.Rows.Add(newRow2);

var query1 = dt.Rows.Cast<DataRow>().Select(e=>(int)e["A"]);
var query2 = dt.Rows.OfType<DataRow>().Select(e=>(int)e["A"]);

这样我们就可以得到看上去两个相同的序列,在这里要注意:MSDN上的说明存在误导,MSDN对于OfType<T>的解释存在偏差,实际上经本人反复敲代码验证,得到的结论是,Cast对序列进行强制转换,一旦转换不成功则抛出异常。OfType则是一旦转换不成功,则不会抛出异常,但是将会得到一个空序列。见下面代码:

var arr1 = new string[] { "1","2","test" };
var arr2 = arr1.Cast<int>();
var arr3 = arr1.OfType<int>();

//通过Cast转换,则会抛出异常
foreach (var i in arr2)
     Console.WriteLine(i.ToString());

//通过OfType转换,有异常但是不会抛出并得到一个空序列
foreach (var i in arr3)
      Console.WriteLine(i.ToString());

Console.Read();

  本文到此,我们已对LINQ涉及的应用有了进一步的了解。学习什么是linq的延迟加载,lambda和linq是否有暧昧关系。以及静态扩展方法对linq的辅助作用。也许你会问既然可以用静态扩展方法替代编写linq,那么二者怎么择取呢,据砖家叫兽提议我们应该先以linq命名空间下的静态扩展方法为主,实在是很复杂的linq表达式,我们再考虑使用linq本身的表达式编写。后续我们将分享学习LINQ更贴近实战应用的知识,linq to dataset, linq to xml, linq to sql, linq to entities.

  感谢您的阅读,如果有说得不对的地方请指正。

来自森大科技官方博客
http://www.cnsendblog.com/index.php/?p=163
GPS平台、网站建设、软件开发、系统运维,找森大网络科技!
http://cnsendnet.taobao.com

原文地址:https://www.cnblogs.com/cnsend/p/12040362.html

时间: 2024-10-03 22:54:23

LINQ入门(中篇)的相关文章

LINQ入门(下篇)

来自森大科技官方博客http://www.cnsendblog.com/index.php/?p=172GPS平台.网站建设.软件开发.系统运维,找森大网络科技!http://cnsendnet.taobao.com 到现在为止你还未触碰LINQ,那进来吧 —— LINQ入门(下篇) 前言 终于来到下篇了,通过上篇,和中篇,我们了解了linq的基本语句,对应linq我们又了解到lambda表达式,静态扩展方法,以及linq的延迟加载的特性,那么在本篇文章中我们将分享学习一下linq对于我们开发中

Asp.net之LINQ入门视频教程

当前位置: 主页 > 编程开发 > Asp.net视频教程 > Asp.net之LINQ入门视频教程 > http://www.xin1234.com/Program/AspnetShiPin/AspNetLINQYmSp/ 1.掌握LINQ中的基本概念 上传日期:2014-09-17 02:57:38 相关摘要: - LINQ有关的语言特性:隐式类型 - 网络不稳定的基本处理方法 - python的编程概念比较全面,比vba来得全面,适用范围也广得多了 2.理解扩展方法Lambd

新注册第一帖----------------------乱码新手自学.net 之Linq 入门篇

作为一个业余开发,断断续续学.net/c#也有不少日子了, 学习过程中,不断忘了学,学了忘,这让我很苦恼. 以前学习过程中,我总是在笔记本中记录下来知识要点,这么久下来,笔记本都写了四五本了. 然而,随着笔记本的增多,自己很快发现,笔记写了跟没写一样:笔记多了就找不到了-- 所以,我觉得还是上博客园写博客,记录自己每天的学习心得. 如果有什么错误的地方,欢迎大神指教,小弟在这给大神跪谢了 ======================================================

Linq之旅:Linq入门详解(Linq to Objects)

示例代码下载:Linq之旅:Linq入门详解(Linq to Objects) 本博文详细介绍 .NET 3.5 中引入的重要功能:Language Integrated Query(LINQ,语言集成查询).通过LINQ,我们可以使用相同API操作不同的数据源.接下来就让我们看看LINQ是什么以及如何使用? 再此之前,需要先了解的相关技术 1. 隐式类型.匿名类型.对象初始化器 1) 隐式类型,使用var关键字创建,C#编译器会根据用于初始化局部变量的初始值推断出变量的数据类型.(不过我个人认

LINQ入门(完结篇)

来自森大科技官方博客http://www.cnsendblog.com/index.php/?p=175GPS平台.网站建设.软件开发.系统运维,找森大网络科技!http://cnsendnet.taobao.com 到现在为止你还未触碰LINQ,那进来吧 -- LINQ入门(完结篇)前 言 各种懒惰,各种拖沓,终究是要动笔写终结篇了,在这个系列的前几篇文章里我们主要学习linq的基础语法以及他对内存数据的操作等,那么本篇文章我们将讨论学习最为大家所熟悉的,也是最受争议的 Linq To SQL

linq入门系列导航

写在前面 为什么突然想起来学学linq呢?还是源于在跟一个同事聊天的时候,说到他们正在弄得一个项目,在里面用到了linq to sql.突然想到距上次使用linq to sql是三年前的事情了.下班回到家,翻看了电脑上面关于linq的笔记,它已经逃得无影无踪了.也怪我,没事瞎折腾电脑,早不知道放哪儿了,隐隐约约记得之前写过这样的笔记.没办法,只能重新整理学习了.这也是临近春节,还是把导航放出来吧,感兴趣的,有想回家给自己充充电的,也可以参考一下. 系列文章 Linq之Lambda表达式初步认识

Linq入门——什么是linq &amp; 扩展方法

一,什么是Linq linq(language integrated Query):语言集成查询: linq包含如下: 对对象的查询,对数据库的查询,对XML的查询. 那么,没有linq前我们是怎样查询的? 先看一个例子: 现在我们要查询大于50的数,: 在没有linq之前,我们的代码时这样的: 使用了linq查询: 首先,从直观上看,代码更加简洁,其次,对于查询部分,接近SQL语句,层次清晰,容易理解: 除了简单高效以为,LINQ的出现解决了很多问题: 1,面向对象与数据访问两个领域长期分裂,

LINQ入门

LINQ 是C#中内置的查询语言,感觉和Sql比较类似,主要包括了LINQ to Object,LINQ to XML,LINQ to SQL为不同类型的数据提供了解决方案.暂时只看了LINQ to Object部分,后面的等我看完再补上. 从简单的数组开始,设有一个数组int[] nums存放者大量数据,现在用LINQ来查找其中>1000的存在,并从大到小排序. var sqlResult = from n in nums where n >1000 orderby descending s

LINQ TO SQL Linq入门基础

我们以一个酒店管理系统的数据库为例子 表结构很简单:GuestInfo(客人信息表),Room(房间表),RoomType(房间类型表),HotelInfo(每个月盈利信息) 整体就是,Room中有个房间类型ID的外键,客人表中有个房间的ID的外键.月盈利信息先不管 准备:新建项目 linq_Ch4控制台程序,新建DB文件夹 右击DB文件夹,添加→新建项 选择LINQ to SQL类,文件名为 Hotel.dbml,添加 打开服务器资源管理器,我的快捷键是Ctrl+W+L 选中这四张表,往已经打