Linq转换操作之OfType,Cast,AsEnumerable,ToLookup源码分析

Linq转换操作之OfType,Cast,AsEnumerable,ToLookup源码分析

一:Tolookup

1. 从方法的注解上可以看到,ToLookup也是一个k,v的形式,那么问题来了,它和Dcitionary有什么区别???

可以看到ToDictionray中的源码是这样的:

foreach (TSource current in source)
{
dictionary.Add(keySelector(current), elementSelector(current));
}

大家有没有看到,这个源码有什么不合实际的地方。。。

如果key重复了,那么这个ToDictionary却对是要报错的。。。。

现在我的要求来了,如果说key重复了,我希望value中的值类型起来。这个就是我们今天要讲的ToLookup要解决的问题。

二:lookup的源码分析

Lookup<TKey, TElement> lookup = new Lookup<TKey, TElement>(comparer);
foreach (TSource current in source)
{
lookup.GetGrouping(keySelector(current), true).Add(elementSelector(current));
}

大家课后可以仔细看一看。

三:OfType转换运算符 (筛选制定类型)

1. 解释: 根据指定类型筛选 System.Collections.IEnumerable 的元素。

     var list = new List<object>() { , "3D",  };

     var array = list.OfType<int>(); //结果是 int 集合

通过demo,我们可以看到OfType就是筛选指定的类型。

很可惜,ILSpy看不到。

四:Cast运算符

1. 解释: 将 System.Collections.IEnumerable 的元素强制转换为指定的类型。

这也告诉我们,ILSpy也不是所有的代码都能看得到。。。

Cast的扩展方法

1. 将 IEnumerable 的元素强制转换为指定的类型。 这句话有强大的误导性

System.Collections.ArrayList fruits = new System.Collections.ArrayList();
fruits.Add("mango");
fruits.Add("apple");
fruits.Add("lemon");

IEnumerable<string> query =
fruits.Cast<string>().OrderBy(fruit => fruit).Select(fruit => fruit);

为什么这个可以cast成功的原因是什么???

string => object => string[cast]

这个过程叫做拆装箱。。。

static void Main(string[] args)
{
var list = new int[] { 10, 20, 30 };

var query = list.Cast<object>().ToList(); //装箱

var myquery = query.Cast<int>().ToList(); //拆箱

var myquery2 = list.Cast<long>().ToList(); //隐式转换 [不可以]
}

可以看得到,Cast并不是我们看到如解释一样那么容易理解。

五:asEnumerable

如果当前的类型没有继承IEumverable这个接口,那么我们可以强制将这个类型转换为继承子

IEnumverable接口的类。

var table = new DataTable();

table.AsEnumerable().Select(i => i.Field<string>(""));

上一篇:XE6 c++builder 设置 font size GetPropInfo SetOrdProp


下一篇:Linq转换操作之ToArray,ToList,ToDictionary源码分析