快速业务通道

LINQ学习笔记:查询是怎么执行的

作者 佚名技术 来源 NET编程 浏览 发布时间 2012-03-14

外部变量

如果查询语句中的Lambda表达式引用了本地变量, 这些变量的值将是查询被执行时候的值, 而不是第一次被捕获时候的值, 例如:

   1: int[] numbers = { 10, 20 };
   2: int factor = 10;
   3: var query = numbers.Select (n => n * factor);
   4: factor = 20;      // Change value
   5: foreach (int n in query)
   6:   Console.Write (n + ",");   // 200,400,

 

当我们使用一个foreach循环创建一个查询的时候这可能会是一个陷阱, 例如:

   1: IEnumerable<char> query = "Not what you might expect";
   2: foreach (char vowel in "aeiou")
   3: {
   4:   char temp = vowel;
   5:   query = query.Where (c => c != temp); //如果使用vowel, 那么将只有’u’会被删除
   6: }

 

延迟执行是如何工作的?

查询操作符通过返回装饰过的序列来提供延迟执行功能.

与传统的集合类( 例如Array或者Linked List)不同的是, 一个装饰过的序列本身并没有自己的数据结构来存储元素, 相反的, 它包装了你在运行时提供的序列, 因此, 它保持了一个永久的依赖. 任何时候你从装饰器中请求数据, 它都会将请求转发到包装过的输入序列中去.

调用Where操作符的时候只是构造了装饰序列, 并保持了输入序列, Lambda表达式和其他参数的引用. 只有当装饰器被枚举的时候输入序列才会被枚举.

例如:

   1: IEnumerable<int> lessThanTen =
   2:       new int[] { 5, 12, 3 }.Where (n => n < 10);

 

当你枚举lessThanTen的时候, 你实际上是在通过where装饰器查询数组. 一个好消息是你可以通过一个C#迭代器来实现装饰序列从而编写你自己的查询操作符. 以下演示了怎么编写你自己的Select方法:

   1: static IEnumerable<TResult> Select<TSource,TResult> (
   2:   this IEnumerable<TSource> source,
   3:   Func<TSource,TResult> selector)
   4: {
   5:   foreach (TSource element in source)
   6:     yield return selector (element);
   7: }

 

此方法是利用了yield return表达式的优点返回了一个迭代器, 以下是一个相同表达的快捷版本:

   1: static IEnumerable<TResult> Select<TSource,TResult> (
   2:   this IEnumerable<TSource> source,
   3:   Func<TSource,TResult> selector)
   4: {
   5:   return new SelectSequence (source, selector);
   6: }

 

凌众科技专业提供服务器租用、服务器托管、企业邮局、虚拟主机等服务,公司网站:http://www.lingzhong.cn 为了给广大客户了解更多的技术信息,本技术文章收集来源于网络,凌众科技尊重文章作者的版权,如果有涉及你的版权有必要删除你的文章,请和我们联系。以上信息与文章正文是不可分割的一部分,如果您要转载本文章,请保留以上信息,谢谢!

分享到: 更多

Copyright ©1999-2011 厦门凌众科技有限公司 厦门优通互联科技开发有限公司 All rights reserved

地址(ADD):厦门软件园二期望海路63号701E(东南融通旁) 邮编(ZIP):361008

电话:0592-5908028 传真:0592-5908039 咨询信箱:web@lingzhong.cn 咨询OICQ:173723134

《中华人民共和国增值电信业务经营许可证》闽B2-20100024  ICP备案:闽ICP备05037997号