linq中的转换操作符

 更新时间:2022年03月10日 11:23:27   作者:.NET开发菜鸟  
这篇文章介绍了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的扩展方法。来看方法的定义:

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

看看下面的例子:

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

二、ToArray操作符

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

public T[] ToArray();

看下面的例子:

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

三、ToDictionary操作符

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

开看下面的例子。

先定义Category类:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConvertOperation
{
    public class Category
    {
        public int Id { get; set; }
        public string CategoryName { get; set; }
        public DateTime CreateTime { get; set; }
    }
}

调用:

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

结果:

注意:

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

看下面的例子:

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

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

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

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

四、ToList操作符

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

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

来看下面的例子:

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

五、ToLookUp操作符

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

public static ILookup<TKey, TSource> ToLookup<TSource, TKey>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector);
public static ILookup<TKey, TElement> ToLookup<TSource, TKey, TElement>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector);
public static ILookup<TKey, TElement> ToLookup<TSource, TKey, TElement>(this IEnumerable<TSource> source, Func<TSource, TKey> keySelector, Func<TSource, TElement> elementSelector, IEqualityComparer<TKey> comparer);
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可以对应多个产品,其定义如下:

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;

namespace ConvertOperation
{
    public class Product
    {
        public int Id { get; set; }
        public int CategoryId { get; set; }
        public string Name { get; set; }
        public double Price { get; set; }
        public DateTime CreateTime { get; set; }
    }
}

在方法中调用:

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

结果:

注意:

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

看下面的例子:

var list1 = listProduct.ToLookup(p => p.CategoryId);
foreach (var item in list1)
{
      Console.WriteLine($"key:{item.Key}");
      foreach (var p in item)
      {
          Console.WriteLine($"Id:{p.Id},CategoryId:{p.CategoryId},Name:{p.Name},CreateTime:{p.CreateTime}");
      }
}

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

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

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

六、Cast操作符

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

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

来看下面的例子:

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

结果:

注意:

  • 1、使用Cast()方法时必须要传入类型参数。
  • 2、序列中的元素必须要能转换为类型 TResult。看下面的例子:
ArrayList arrayList = new ArrayList();
arrayList.Add(1);
arrayList.Add("2");
arrayList.Add(3);
//非泛型转换成泛型
var list = arrayList.Cast<int>();
foreach (var item in list)
{
     Console.WriteLine(item);
}

程序运行结果:

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • .net core日志系统相关总结

    .net core日志系统相关总结

    多年的经验,日志记录是软件开发的重要组成部分。没有日志记录机制的系统不是完善的系统。在开发阶段可以通过debug附件进程进行交互调试,可以检测到一些问题,但是在上线之后,日志的记录起到至关重要的作用。本文讲解下日志系统的相关使用
    2021-06-06
  • 比较完整的 asp.net 学习流程

    比较完整的 asp.net 学习流程

    好多朋友想学习后台编程语言,但请注意的事,学习后台是个循序渐进的过程,不可能一下就到位,其实不只是asp.net其它的编程语言都需要下面的一些知识。
    2009-06-06
  • asp.net小谈网站性能优化

    asp.net小谈网站性能优化

    随着公司的栏目越来越多,数据量的庞大。网站的性能就越崭露头角。
    2011-02-02
  • asp.net使用母版页中使用ajax脚本取数据

    asp.net使用母版页中使用ajax脚本取数据

    因母版页继承自UserControl,我们无法像正常页面那样使用Jquey或Ajax的PageMethods等无刷新方法取数据。不过可以使用ajax提供的Sys.Net.WebRequest来解决这一问题。
    2010-09-09
  • 使用CustomValidator自定义验证控件检查是否有对ListBox控件选择

    使用CustomValidator自定义验证控件检查是否有对ListBox控件选择

    在网页前端处,我们放置ListBox控件,在数据提交前,检查用户是否有对此控件进行选择?小编将介绍使用Javascript与CustomValidator自定义验证控件来检查感兴趣的朋友可以了解下
    2013-01-01
  • 理解ASP.NET Core 启动类(Startup)

    理解ASP.NET Core 启动类(Startup)

    这篇文章主要介绍了ASP.NET Core 启动类(Startup),文中运用代码讲解相关知识非常详细,感兴趣的小伙伴可以参考一下
    2021-09-09
  • ASP.NET Web应用程序出现Maximum request length exceeded报错的解决方法

    ASP.NET Web应用程序出现Maximum request length 

    ASP.NET Web应用中导出数据时出现500-Internal Server Error,原因是客户端请求长度超过了服务器配置的最大限制,解决方法在web.config增加maxRequestLength属性,单位为字节(Byte),本文介绍ASP.NET Web应用程序出现Maximum request length exceeded报错的原因,一起看看吧
    2024-12-12
  • ASP.NET State service状态服务的问题解决方法

    ASP.NET State service状态服务的问题解决方法

    每次重启机器以后,在.NET2005上跑Web程序老是遇到这样的问题
    2008-11-11
  • .net中捕捉全局未处理异常的三种方式示例

    .net中捕捉全局未处理异常的三种方式示例

    这篇文章主要给大家介绍了关于.net中捕捉全局未处理异常的三种方式,分别是Page_Error处理页面级未处理异常、通过HttpModule来捕获未处理的异常以及通过Global中捕获未处理的异常,需要的朋友可以参考下
    2018-06-06
  • 透过ashx看浏览器服务器运行本质(图解)

    透过ashx看浏览器服务器运行本质(图解)

    一般处理程序(HttpHandler):是一个实现System.Web.IHttpHandler接口的特殊类。任何一个实现了IHttpHandler接口的类是作为一个外部请求的目标程序的前提,感兴趣的朋友可以了解下或许有所帮助
    2013-01-01

最新评论