当前位置: 首页 > 图文教程 > 网络编程 > ASP.NET > LINQ学习笔记:子查询和延迟执行

ASP.NET
赫赫大名的A*寻路算法(vb.net版本)
asp.net(c#)下Jmai去说明 使用与下载
[原创]完美解决Could not load file or assembly ''AjaxPro.2'' or one of its dependencies. 拒绝访问。
asp.net下gridview 批量删除的实现方法
用CSS实现图片倾斜 只支持IE
.net get set用法小结
vs 不显示行号的操作方法
ASP.NET页面进行GZIP压缩优化的几款压缩模块的使用简介及应用测试!(附源码)
ASP.Net不执行问题一解
asp.net 无限分类
让VS2008对JQuery语法的智能感知更完美一点
扩展方法ToJSON() and ParseJSON()
asp.net下PageMethods使用技巧
Linq to SQL Delete时遇到问题的解决方法
实现ASP.NET多文件上传程序代码
ASP.NET AJAX 1.0 RC开发10分钟图解
asp.net get set用法
ASP.NET下使用WScript.Shell执行命令
asp.net2.0实现邮件发送(测试成功)
Asp.net 无限级分类实例代码

ASP.NET 中的 LINQ学习笔记:子查询和延迟执行


出处:互联网   整理: 软晨网(RuanChen.com)   发布: 2009-09-28   浏览: 58 ::
收藏到网摘: n/a

子查询

子查询是一个包含了另外一个查询的Lambda表达式的查询. 以下的例子使用了一个子查询来针对篮球明星的last name排序:

 1: string[] players = { "Tim Ducan", "Lebrom James", "Kobe Byrant" };
 2: IEnumerable<string> q = players.OrderBy (m => m.Split().Last());

在这其中, Last是一个子查询, q则代表了一个外部查询.

在子查询中, 你可以在Lambda表达式的右边使用任何可行的C#表达语法. 子查询只是一个简单的C#表达式, 这意味着所有适用于子查询的规则都可以推导到Lambda表达式上.

以下的查询取得一个字符数组中所有满足长度等于最小长度的字符序列:

 1: string[] names = { "James","Jack","Landy","C.Y","Jay" };
 2: IEnumerable<string> q = names
 3: .Where (n => n.Length ==
 4: names.OrderBy (n2 => n2.Length)
 5: .Select (n2 => n2.Length).First( )
 6: );
 7: foreach(var s in q)
 8: {
 9: Console.WriteLine(s); //C.Y , Jay
 10: }

对于子查询, 可以引用到外部的Lambda参数或者是迭代变量(在复合查询中). 例如上述的例子中, 如果OrderBy使用的表达式改为(n => n.Length)而不是用n2的话将会得到一个错误信息:

A local variable named ‘n’ cannot be declared in this scope because it would give a different meaning to ‘n’, which is already used in a ‘parent or current’ scope to denote something else.

针对这个例子, 我们可以看到对应的复合查询写法:

 1: IEnumerable<string> q =
 2: from n in names
 3: where n.Length ==
 4: (from n2 in names
 5: orderby n2.Length
 6: select n2.Length).First( )
 7: select n;

外部迭代变量n在子查询范围内是可见的, 因此我们不能将它重用为子查询内部的迭代变量.

子查询会在对应的Lambda表达式被执行的时候来执行, 其执行取决于外部查询, 也可以说是由外到里来处理的. 本地查询完全遵循这个模型, 但是解释型查询(例如LINQ to SQL)则仅仅是概念上遵循而已.

之前的查询我们还可以使用一种更加简洁的写法:

 1: IEnumerable<string> q =
 2: from n in names
 3: where n.Length ==
 4: names.OrderBy (n2 => n2.Length).First().Length
 5: select n;

如果使用Min聚合函数, 还可以进一步简化:

 1: IEnumerable<string> q =
 2: from n in names
 3: where n.Length == names.Min (n2 => n2.Length)
 4: select n;

实际上, 由于n2.Length在外部查询循环的时候每次都会重新计算, 这在某些情况下可能会引起效率问题, 避免这个问题, 我们可以将子查询分离出来:

 1: int len = names.Min (n => n.Length);
 2:  
 3: IEnumerable<string> query = from n in names
 4: where n.Length == len
 5: select n;

子查询和延迟执行

在子查询中的返回单一元素或者聚合类操作符, 例如first或者Count, 并不会强制外部查询立即执行, 也就说外部查询依然拥有延迟执行的能力. 这是因为子查询是被间接调用的 – 如果是本地查询则是通过代理(delegate), 如果是解释性查询则是通过表达树(expression tree).

一个有趣的现象是当你的子查询中包含一个Select表达式的时候, 如果是本地查询, 你实际上是将其发散成一序列的查询 – 并且每一个都拥有延迟执行的能力. 这个影响是透明的, 因为它可以显著提高效率. 待续!