linq操作符:转换操作符

这些转换操作符将集合转换成数组:IEnumerable、IList、IDictionary等。转换操作符是用来实现将输入对象的类型转变为序列的功能。名称以"As"开头的转换方法可更改源集合的静态类型但不枚举(延迟加载)此源集合。名称以"To"开头的方法可枚举(即时加载)源集合并将项放入相应的集合类型。

一、AsEnumerable操作符

所有实现了IEnumerable<T>接口的类型都可以调用此方法来获取一个IEnumerable<T>集合。AsEnumerable操作符可以将一个类型为IEnumerable<T>的输入序列转换成一个IEnumerable<T>的输出序列,其主要用于将一个实现了IEnumerable<T>接口的对象转换成一个标准的IEnumerable<T>接口对象。在Linq中,不同领域的Linq实现都有自己专属的操作符。

例如:IQueryable<T>通常是Linq to SQL的返回类型,当我们之间在上面调用Where<>方法时,调用的是Linq to SQL的扩展方法,因此有时候我们需要转换为标准的IEnumerable<T>,然后在调用Linq to OBJECT的扩展方法。来看方法的定义:

1 public static IEnumerable<TSource> AsEnumerable<TSource>(this IEnumerable<TSource> source)

 看看下面的例子:

1 DataTable dt = new DataTable();
2 // 将dt先使用AsEnumerable()操作符进行转换,然后在调用Linq to Object 的where方法
3 var list= dt.AsEnumerable().Where(p => p.Name.length > 0);

 二、ToArray操作符

ToArray操作符可以在IEnumerable<T>类型的任何派生对象上调用,返回值为T类型的数组。其方法定义如下:

1 public T[] ToArray();

 看下面的例子:

1 List<int> list = new List<int>();
2 list.Add(1);
3 list.Add(3);
4 list.Add(4);
5 // 将List<int>类型的集合转换成int[]数组
6 int[] intArray = list.ToArray();

三、ToDictionary操作符

ToDictionary操作符根据指定的键选择器函数,从IEnumerable<T>创建一个Dictionary<TKey,TValue>。

开看下面的例子。

先定义Category类:

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5 using System.Threading.Tasks;
 6 
 7 namespace ConvertOperation
 8 {
 9     public class Category
10     {
11         public int Id { get; set; }
12         public string CategoryName { get; set; }
13         public DateTime CreateTime { get; set; }
14     }
15 }

 调用:

 1 List<Category> listCategory = new List<Category>()
 2 {
 3         new Category(){ Id=1,CategoryName="计算机",CreateTime=DateTime.Now.AddYears(-1)},
 4         new Category(){ Id=2,CategoryName="文学",CreateTime=DateTime.Now.AddYears(-2)},
 5         new Category(){ Id=3,CategoryName="高校教材",CreateTime=DateTime.Now.AddMonths(-34)},
 6         new Category(){ Id=4,CategoryName="心理学",CreateTime=DateTime.Now.AddMonths(-34)}
 7 };
 8 var dict= listCategory.ToDictionary(c => c.Id, c => c.CategoryName);
 9 foreach(var item in dict)
10 {
11         Console.WriteLine($"key:{item.Key},value:{item.Value}");
12 }

 结果:

注意:

1、如果省略ToDictionary()方法的第二个参数(值选择函数),那么value将会保存一个类别对象。看下面的例子:

1 var dict = listCategory.ToDictionary(c=>c.Id);
2 foreach (var item in dict)
3 {
4     Console.WriteLine($"key:{item.Key},Id:{dict[item.Key].Id},CategoryName:{dict[item.Key].CategoryName},CreateTime:{dict[item.Key].CreateTime}");               
5 }

在程序运行时打断点,查询value的类型:

从截图中可以看出:这时value的类型是Category类型。其输出结果如下:

2、如果key值为null或者出现重复的key,那么将会导致程序抛出异常。(字典的key值不可以是重复的)

四、ToList操作符

ToList操作符可以在IEnumerable<T>类型的任何派生对象上使用,返回值是List<T>类型的集合。其定义如下:

1 public static List<TSource> ToList<TSource>(this IEnumerable<TSource> source);

 来看下面的例子:

1 int[] intArray = { 1, 2, 3, 56, 78, 34 };
2 List<int> list = intArray.ToList();

 五、ToLookUp操作符

ToLookUp操作符将创建一个LookUp<TKey,TElement>对象,这是一个one-to-many的集合,一个key可以对应多个value值。其方法定义如下:

1 public static ILookup<TKey, TSource> ToLookup<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector);
2 public static ILookup<TKey, TElement> ToLookup<TSource, TKey, TElement>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, 
Func<TSource, TElement> elementSelector); 3 public static ILookup<TKey, TElement> ToLookup<TSource, TKey, TElement>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector,
Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer); 4 public static ILookup<TKey, TSource> ToLookup<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, IEqualityComparer<TKey> comparer);

从方法定义中可以看出:ToLookUp的value值的类型和集合中元素的类型一致。如果一个key对应多个value值,那么value会是TSource类型的集合。 

来看下面的例子。

先定义Product类,Product类中的分类ID可以对应多个产品,其定义如下:

 1 using System;
 2 using System.Collections.Generic;
 3 using System.Linq;
 4 using System.Text;
 5 using System.Threading.Tasks;
 6 
 7 namespace ConvertOperation
 8 {
 9     public class Product
10     {
11         public int Id { get; set; }
12         public int CategoryId { get; set; }
13         public string Name { get; set; }
14         public double Price { get; set; }
15         public DateTime CreateTime { get; set; }
16     }
17 }

 在方法中调用:

 1 List<Product> listProduct = new List<Product>()
 2 {
 3       new Product(){Id=1,CategoryId=1, Name="C#高级编程第10版", Price=100.67,CreateTime=DateTime.Now},
 4       new Product(){Id=2,CategoryId=1, Name="Redis开发和运维", Price=69.9,CreateTime=DateTime.Now.AddDays(-19)},
 5       new Product(){Id=3,CategoryId=2, Name="活着", Price=57,CreateTime=DateTime.Now.AddMonths(-3)},
 6       new Product(){Id=4,CategoryId=3, Name="高等数学", Price=97,CreateTime=DateTime.Now.AddMonths(-1)},
 7       new Product(){Id=5,CategoryId=6, Name="国家宝藏", Price=52.8,CreateTime=DateTime.Now.AddMonths(-1)}
 8 };
 9 var list = listProduct.ToLookup(p => p.CategoryId, p => p.Name);
10 foreach (var item in list)
11 {
12       Console.WriteLine($"key:{item.Key}");
13       foreach (var p in item)
14       {
15           Console.WriteLine($"value:{p}");
16       }
17 }

结果:

注意:

1、如果省略ToLookUp()方法的第二个参数(值选择函数),那么value将会保存一个类别对象。看下面的例子: 

1 var list1 = listProduct.ToLookup(p => p.CategoryId);
2 foreach (var item in list1)
3 {
4       Console.WriteLine($"key:{item.Key}");
5       foreach (var p in item)
6       {
7           Console.WriteLine($"Id:{p.Id},CategoryId:{p.CategoryId},Name:{p.Name},CreateTime:{p.CreateTime}");
8       }
9 }

 程序运行时打断点,查看value值的类型:

从上面的截图中能够看出:这时value的类型是Product类型。运行结果如下:

2、ToLookUp和GroupBy操作很相似,只不过GroupBy是延迟加载的,ToLookUp是立即加载的。

六、Cast操作符

Cast操作符用于将一个类型为IEnumerable的集合对象转换为IEnumerable<T>类型的集合对象。也就是非泛型集合转成泛型集合,因为在Linq to OBJECT中,绝大部分操作符都是针对IEnumerable<T>类型进行的扩展方法。因此对非泛型集合并不适用。来看方法定义:

1 public static IEnumerable<TResult> Cast<TResult>(this IEnumerable source);

 来看下面的例子:

 1 ArrayList arrayList = new ArrayList();
 2 arrayList.Add(1);
 3 arrayList.Add(2);
 4 arrayList.Add(3);
 5 //非泛型转换成泛型
 6 var list = arrayList.Cast<int>();
 7 foreach (var item in list)
 8 {
 9      Console.WriteLine(item);
10 }

 结果:

注意:

1、使用Cast()方法时必须要传入类型参数。

2、序列中的元素必须要能转换为类型 TResult。看下面的例子:

 1 ArrayList arrayList = new ArrayList();
 2 arrayList.Add(1);
 3 arrayList.Add("2");
 4 arrayList.Add(3);
 5 //非泛型转换成泛型
 6 var list = arrayList.Cast<int>();
 7 foreach (var item in list)
 8 {
 9      Console.WriteLine(item);
10 }

程序运行结果:

 

posted @ 2018-07-15 09:22  .NET开发菜鸟  阅读(751)  评论(0编辑  收藏  举报