范型在c#编程中经常使用,而经常用list 去存放实体集,因此会设计到对list的各种操作,比较常见的有对list进行排序,查找,比较,去重复。而一般的如果要对list去重复如果使用linq distinct方式,会遇到一些坑爹的问题,发现结果集中还是存在重复数据,原因是使用这种方法是对对象的引用去重复,并不满足我们的需求。因此本文通过c#代理的方式实现对list distinct操作。
先介绍一下对list去重复传统的方法,代码如下:
List<ReviewersReport> reportList= GetReportList(); for ( int i = 0 ; i < reportList.Count; i++ ) { for ( int j = i + 1 ; j < reportList.Count; j++ ) { if (reportList[i].Equals(reportList[j])) { reportList.RemoveAt(reportList.LastIndexOf(reportList[i])); j -- ; } } }
通过这种方式对list 实现distinct操作显然比较麻烦,如果还有其他的list实体集也需要实现类似的功能,那我们就会为代码的可重用性担心了。
下面使用简单高效的方式去实现list的distinct功能,也是本文推荐的方式了
先创建一个Compare类,如下:
public delegate bool EqualsComparer<T> (T x, T y); public class Compare<T> : IEqualityComparer<T> { private EqualsComparer<T> _equalsComparer; public Compare(EqualsComparer<T> equalsComparer) { this ._equalsComparer = equalsComparer; } public bool Equals(T x, T y) { if ( null != this ._equalsComparer) return this ._equalsComparer(x, y); else return false ; } public int GetHashCode(T obj) { return obj.ToString().GetHashCode(); } }
这里在构造器中传递一个delegate,调用者可以在这个delegate定义比较规则,这样具有了极大的灵活性,我们可以注意到Compare实现了IEqualityComparer接口来自定义比较对象,判断两个对象是否相等。使用方式如下:
ist<ReviewersReport> requestList = Get RequestList (); requestList =requestList.Distinct( new Compare<Requestor>((x, y) => ( null != x && null != y) && (x.RequestorName.Equals(y.RequestorName)))).ToList();
用这种方式大大的扩展了比较器的使用范围,增加了代码的可重用性,可以适用于任何对象的比较。
比较后我们可以对起进行排序,使代码也很简洁。
requestList.Sort( delegate (Requestor r1, Requestor r2) { return r1.RequestorCSL.CompareTo(r2.RequestorCSL); });
希望对园友们有所帮助。。。