C# GroupBy使用
2021-06-16 21:04
标签:var 就是 googl 今天 需要 prope 内存 entity 存在 今天在公司做一个需求的时候,写的是面条代码,一个方法直接从头写到尾,其中用到了 但是公司最近推行Clean Code,要让代码有可读性。且作为一个有追求的程序员,肯定是不能写面条代码的,要对代码进行拆分。 重构前 个人对于短的Linq比较习惯于用方法而不是用关键字的那种写法。 一开始这样写是没问题的,但是重构的时候问题就来了:这个 重构以后这个 我们知道 但是这个 我没有指定 但是后来我发现这样有问题, 发现这个问题后,我仔细思考了一下,大致猜到了问题出在哪里。 虽然我没有看匿名类型反编译生成后的 这样子肯定是不行的,要解决引用类型的相等判断问题。 根据猜测,我写了一个Sample程序最小化的重现了这个问题: 这时候输出结果是 Age:10,Class:University1 将 Age:10,Class:University1 解决问题方式有几种。 最简单,就是直接将 但是这样有个问题, 虽然实际情况不一定会出现内存异常什么的,但是总归是改变了一些东西,存在隐患。 第一种方式被我自己否决后,于是打开了Google搜了一下,在StackOverflow和MSDN以及查看 但不管是默认的 第二种方法就是我们在类型上重载 可以实现 但是不论如何,一定要重载 修改后 第三种就是传一个 代码如下: 这种相对于第二种方式,最大的区别在于不用侵入实体类添加代码,但是原理是类似的。 本文是在C#开发过程中碰到的一个 了解其分组原理后,通过实现 C# GroupBy使用 标签:var 就是 googl 今天 需要 prope 内存 entity 存在 原文地址:https://www.cnblogs.com/Weilence/p/10343036.html起因
GroupBy
,且GroupBy
的KeySelector
是多个属性而不是单个属性。GroupBy
大概是这样子的:var groups = data.GroupBy(m => new { m.PropertyA, m.PropertyB})
groups
是什么类型?groups
是要作为参数进入到别的方法中的,方法签名显然是不能用var
做类型推导,必须指定确定的类型。GroupBy
出来的东西是个泛型的东西,签名是IEnumerable
,这个TSource
类型是没问题,我没有对Source
做修改,就是data
本身的类型。Key
就有问题了。Key
的类型,这里应该是匿名类型,于是定义了一个类型承接Key
,代码变成了:class EntityKey
{
public int PropertyA { get set; }
public string PropertyB { get set; }
}
......
var groups = data.GroupBy(m => new EntityKey { PropertyA = m.PropertyA, PropertyB = m.PropertyB});
GroupBy
指定的Key
失效了。也就是说,groups
的分组数量与data
的长度一致,每一个group
里面只有一个对象。分析
GroupBy
这种东西,判断两个对象是不是一个分组,必然用到了相等判断。IL
代码,不知道之前用的是怎么做的Key相等判断,但是引用类型的肯定是直接用对象的HashCode
做判断。重现
class Program
{
static void Main(string[] args)
{
var list = new List
Id=1,Name=Cat,Age=10,Class=University1
Id=2,Name=Dog,Age=10,Class=University1
Age:10,Class:University2
Id=3,Name=Pig,Age=10,Class=University2
Age:12,Class:University1
Id=4,Name=Fish,Age=12,Class=University1new {m.Age, m.Class}
替换为new StudentKey {Age = m.Age, Class = m.Class}
,结果却变成了
Id=1,Name=Cat,Age=10,Class=University1
Age:10,Class:University1
Id=2,Name=Dog,Age=10,Class=University1
Age:10,Class:University2
Id=3,Name=Pig,Age=10,Class=University2
Age:12,Class:University1
Id=4,Name=Fish,Age=12,Class=University1Id=1
和Id=2
变成了两组。解决问题
第一种
StudentKey
从class
变成struct
。class
是堆内存,struct
是栈内存。第二种
GroupBy
源码之后,得到了GroupBy
的运行原理。GroupBy
在没有传comparer
的时候,会创建一个基于当前TSource
类型的默认的comparer
。comparer
还是我们自己传的comparer
,都会调用Equals
和GetHashCode
两个方法,所以我们需要重载这两个方法。Equals
和GetHashCode
两个方法。IEquatable
使用下面的代码,也可以不实现接口,使用重载的Equals
方法。GetHashCode
。StudentKey
如下class StudentKey : IEquatable
第三种
comparer
给GroupBy
参数,实现一个IEqualityComparer
。list.GroupBy(m => new StudentKey {Age = m.Age, Class = m.Class}, new StudentKeyComparer());
......
class StudentKeyComparer: IEqualityComparer
总结
GroupBy
的分组的Key
失效的问题。Equals
和GetHashCode
或者传入自定义的comparer
,解决GroupBy
的分组Key
失效的问题。