一.查询非泛型集合
1.问题起源
LINQ to object在设计时,是配合IEnumerable<T>接口的泛型集合类型使用的,例如字典、数组、List<T>等,但是对于继承了IEnumerable的非泛型集合如何处理,例如ArrayList。
2.解决办法
上源码:
ArrayList mArrayList = new ArrayList() {"222","1dsadsad","12w1212","1212e12esadq" }; var query = from m in mArrayList.Cast<string>() where m.Length > 3 select m; foreach (var item in query) { Console.WriteLine(item.ToString()); } Console.Read();
Cast操作符是解决此问题的关键。
public static IEnumerable<T> Cast<T>( this IEnumerable source);
解析:
1.从上述代码可以看出,Cast是一个扩展的方法,返回值是IEumerable<T>,Cast的原对象是IEnumerable,针对实现了IEnumerable接口的对象进行扩展。
2.Cast操作符将源序列中的每个元素转型为某一指定的类型:T
3.Cast操作符通过一次遍历源序列中的元素来返回一个可被枚举的对象,当其他代码开始遍历Cast操作的返回结果时,Cast操作符同时开始遍历源序列,转换为T类型,依次返回。
3.OfType
除了Cast操作符之外,我们还可以使用OfType操作符,差别在于OfType操作符只会返回源序列中属于某种特定类型的元素,例如:
ArrayList中同时包含有Book和Student两种不同的类型的对象时,调用ArrayList.OfType<Book>() 将只会返回Book对象。
二.多个分组条件
1.概述
为了实现这个目标,我在group子句中需要指定一个匿名类型
var query=from book in Samp.Books group book by new { book.Publish, book.Title }
解析:
上述查询的是一个集合,其中的每一个元素都是一个分组,每个分组都会包含一个Key(即该匿名类型的一个实例),以及满足该Key的图书集合。
2.再次巩固
上源码:
var query2 = from book in SampleData.booksArray group book by new { book.Publisher, book.Title } into BookGroup select new { publishName=BookGroup.Key.Publisher.Name, title=BookGroup.Key.Title }; foreach (var item in query2) { Console.WriteLine(item.publishName+"; "+item.title); }
这里引入into的目的在于接下来select 和其他子句进行使用,into关键词后的BookGroup变量包含了该组的Key,以及该组中的元素。
Key表示分组的依据,每组中包含的元素则可以通过变量BookGroup变量(该变量实现了IEnumerable<T>接口)得到
源代码: LINQ查询非泛型集合和分组查询测试源码
本文为学习《LINQ实战》总结。