17网站一起做网店普宁,网站做第三方登录,广东网站设计网站,国外专业做集装箱别墅网站背景说明
在EF等ORM框架中需要以List实体类的方式对数据进行大量操作#xff0c;其中免不了对一些数据进行去重复#xff0c;而C#中IEnumerable.Distinct()便提供了这一功能。只是对刚开始接触的新人来说比价抽象难以接受#xff0c;本文会对这一功能进行简要说明#xff…背景说明
在EF等ORM框架中需要以List实体类的方式对数据进行大量操作其中免不了对一些数据进行去重复而C#中IEnumerable.Distinct()便提供了这一功能。只是对刚开始接触的新人来说比价抽象难以接受本文会对这一功能进行简要说明如果有更好的实现方式也请大家畅所语言。
在写本文时本人也在网上搜索了很多相关资料其中有几篇比较有参考价值也是重点本文也是基于这几篇文章提供的代码进行优化和整理
1、用泛型委托实现IEqualityComparer接口https://blog.csdn.net/honantic/article/details/51595823
2、Distinct的多条件查询https://blog.csdn.net/lishuangquan1987/article/details/76096022
3、IEqualityComparer中的Equal()和GetHashCode()https://www.cnblogs.com/xiaochen-vip8/p/5506478.html
我们要做的是实现IEqualityComparer()接口而且必须要用泛型因为我们希望这个功能是可以对所有实体类实现的。其中对哈希值的了解可以参考第三条链接可以简单的概括为哈希值反应的是对象在内存中的地址只有地址相同的对象才能激活IEqualityComparer中的Equal()方法Equal()可以根据自己的需求而实现。话不多说代码如下 /// summary/// 用委托实现IEqualityComparerT接口/// /summary/// typeparam nameT目标类型/typeparampublic class ListComparerT : IEqualityComparerT{public FuncT, T, bool EqualsFunc;public FuncT,int GetHashCodeFunc;public ListComparer(FuncT,T,bool Equals, FuncT,int GetHashCode){this.EqualsFunc Equals;this.GetHashCodeFunc GetHashCode;}public ListComparer(FuncT, T, bool Equals) : this(Equals, t 0){}public bool Equals(T x, T y){if (this.EqualsFunc ! null){return this.EqualsFunc(x, y);}else{return false;}}/// summary/// 获取目标对象的哈希值,只有返回相同的哈希值才能运行Equals方法/// /summary/// param nameobj获取哈希值的目标类型对象/param/// returns返回哈希值/returnspublic int GetHashCode(T obj){if (this.GetHashCodeFunc ! null){return this.GetHashCodeFunc(obj);}else{return 0;}}}
以上代码中默认哈希值是相同的我们开始看看使用效果代码如下 static void Main(string[] args){ListPhone PhoneLists new ListPhone(){new Phone { Country 中国, City 北京, Name 小米 },new Phone { Country 中国,City 北京,Name 华为},new Phone { Country 中国,City 北京,Name 联想},new Phone { Country 中国,City 台北,Name 魅族},new Phone { Country 日本,City 东京,Name 索尼},new Phone { Country 日本,City 大阪,Name 夏普},new Phone { Country 美国,City 加州,Name 苹果},new Phone { Country 美国,City 华盛顿,Name 三星}};var Lists PhoneLists.DistinctPhone();foreach (var list in Lists){Console.WriteLine(list.Country - list.City - list.Name);}Console.Read();}
在Distinct()方法没有任何参数的情况下运行后如下图所示 我们可以看到好像并没有任何效果但是其实是有效果的因为每个Phone实体类对象在内存中的地址是不一样的 Distinct()方法默认筛选出所有内存地址不一样的实体类对象。
接下去需求改变我们希望得出总共有多少个不同的countrycountry相同的数据随便返回其中一个就行代码如下所示 static void Main(string[] args){ListPhone PhoneLists new ListPhone(){new Phone { Country 中国, City 北京, Name 小米 },new Phone { Country 中国,City 北京,Name 华为},new Phone { Country 中国,City 北京,Name 联想},new Phone { Country 中国,City 台北,Name 魅族},new Phone { Country 日本,City 东京,Name 索尼},new Phone { Country 日本,City 大阪,Name 夏普},new Phone { Country 美国,City 加州,Name 苹果},new Phone { Country 美国,City 华盛顿,Name 三星}};var Lists2 PhoneLists.DistinctPhone(new ListComparerPhone((x,y) x.Country.Equals(y.Country)));foreach (var list in Lists){Console.WriteLine(list.Country - list.City - list.Name);}Console.Read();}
我们对country字段进行去重得到的结果如下图所示 再接下去需求又变我们要筛选出有多少不同的国家和城市这意味着要对country和city两个字段进行去重代码如下 static void Main(string[] args){ListPhone PhoneLists new ListPhone(){new Phone { Country 中国, City 北京, Name 小米 },new Phone { Country 中国,City 北京,Name 华为},new Phone { Country 中国,City 北京,Name 联想},new Phone { Country 中国,City 台北,Name 魅族},new Phone { Country 日本,City 东京,Name 索尼},new Phone { Country 日本,City 大阪,Name 夏普},new Phone { Country 美国,City 加州,Name 苹果},new Phone { Country 美国,City 华盛顿,Name 三星}};var Lists PhoneLists.DistinctPhone(new ListComparerPhone((x, y) x.Country.Equals(y.Country) x.City.Equals(y.City)));foreach (var list in Lists){Console.WriteLine(list.Country - list.City - list.Name);}Console.Read();}
执行结果如下图所示 可以看到已经达到了多字段的去重复效果即便遇到需要去重复多个字段也可以实现以上为个人拙见。