在.NET Framework 2.0 引进泛型之前,都是使用System.Collections namespace命名空间下的集合存储对象。在.NET 2.0中,许多集合类纷纷实现了IEnumerable<T>接口而泛型化。但是,还有相当部分的类没有实现IEnumerable的泛型版 本。
LINQ能实现查询泛型对象或者实现了IEnumerable<T>接口的对象。然而,诸如ArrayList这样的非泛型集合并没有实现IEnumerable<T>接口。接下来,让我们看看,在这种情况下,如何使用LINQ查询非泛型集合。
C#
如上面的代码所示,我们声明Cars对象并将其填充到ArrayList集合当中。并且使用类似LINQ查询泛型集合的方法来查询这个ArrayList集合。你认为结果会如何? 通过显式声明变量类型进行查询,你可以将集合中的每个对象转换成指定的对象。 C#
2{
3public string CarMake { get;set;}
4public string CarModel { get; set; }
5public int Year { get; set; }
6}
7
8
9class Program
10{
11static void Main(string[] args)
12{
13 ArrayList carList = new ArrayList();
14 carList.Add(new Cars
15 {
16 CarMake="BMW", CarModel="BMW Art", Year=1978
17 });
18 carList.Add(new Cars
19 {
20 CarMake = "BMW", CarModel = "Coupe", Year = 1982
21 });
22 carList.Add(new Cars
23 {
24 CarMake = "Renault", CarModel = "Alpine", Year = 1972
25 });
26 carList.Add(new Cars
27 {
28 CarMake = "Porsche", CarModel = "Maisto", Year = 1976
29 });
30
31 var carQuery = from car in carList
32 where car.CarMake == "BMW"
33 select car;
34}
当 然,上面的代码并没有通过编译,原因是ArrayList并没有实现IEnumerable<T>接口导致不能使用LINQ进行查询。那么, 是否我们能妄下结论说LINQ不能查询非泛型集合呢?那么那些返回ArrayList对象的类库将何去何从?我们是否能巧妙地处理这些问题或者使用能返回 泛型集体的方法代替呢?在不改变特性的前提下,让我们看看三个解决问题的小技巧。
方法一,显式指定变量类型,该类型与集合中的对象类型必须一致。
2 where car.CarMake == "BMW"
3 select car;
注意我们将转变量car换成Cars
方法二,使用Cast方法
Cast方法可以将(实现IEnumerable)接口的)非泛型集合返回IEnumerable<T>接口对象。一旦我们得到IEnumerable<T>接口对象,我们就可以使用LINQ查询了。
C#
注意,方法一中的所介绍的方法与这种方法相似。
方法三,除了Cast运算符,我们也可以使用OfType
OfType可以过滤集合中相同类型的元素,如果你的集合中包含了不同类型的元素。这个方法得到指定类型的元素对象。
C#
3种方法的全部代码如下所示
C#