C# distnict去重

  在C#中,使用distinct的时候,会发现有些情况不好使,比如我是泛型集合的时候,发现没去掉重复。这是因为对于引用类型,distinct里的默认比较器比较的是其引用地址,程序中集合里的每一个元素都是个新的实例,引用地址都是不同的,所以不会被作为重复记录删除掉。因此我们需要自定义比较器进行去重,实现指定条件去重复。

  本章主要讲解如何设计比较器。我们先来看看第一版:

public class JuryComparer: IEqualityComparer<T_S_SalesPointModel>
    {
        public bool Equals(T_S_SalesPointModel x, T_S_SalesPointModel y)
        { 
            return x.ClientCode.Equals(y.ClientCode);
        }

        public int GetHashCode(T_S_SalesPointModel obj)
        {
            return 0;
        }
    }

  使用的时候很简单,.Distinct(new JuryComparer()),在distinct里实例化这个参数传入即可。

  但是很快我们就发现一个问题,我们别的业务里也需要用到这个,如果作为一个公共方法,显然我是不能把model指定死的,所以有了第二版:

    public class JuryComparer<T> : IEqualityComparer<T>
    {
        public bool Equals(T x, T y)
        {
            dynamic dx = x;
            dynamic dy = y; 
            return dx.ClientCode.Equals(dy.ClientCode);
        }

        public int GetHashCode(T obj)
        {
            return 0;
        }
    }

  通过泛型T我们可以在方法内部不指定具体类型,这样就方法就能自适应不同的model了。

  然鹅,今天遇到一种情况,即,我们自定义的方法中,是通过ClientCode去做比较的,如果外部model中不存在这个Code,那我们就无法比较,因为方法会报错,提示没有这个属性。也就是说我们的通用方法除了model需要通用以外,这个属性,应该从外部传进来,那么如何实现呢,请看第三版:

public class JuryComparer<T> : IEqualityComparer<T>
    {
        private string _attribute;
        public JuryComparer(string attribute)
        {
            _attribute = attribute;
        }
        public bool Equals(T x, T y)
        {
            dynamic dx = x;
            dynamic dy = y;
            var vx = dx.GetType().GetProperty(_attribute).GetValue(dx, null);
            var vy = dy.GetType().GetProperty(_attribute).GetValue(dy, null);
            return vx.Equals(vy);
        }

        public int GetHashCode(T obj)
        {
            return 0;
        }
    }

  通过构造函数传参,我们将外部参数赋值给内部私有变量,最后在比较方法里, 我们通过反射获取属性实现自适应。这样外部就能通过这种方式实现调用:

  .Distinct(new JuryComparer<TreeModel>("value"))

  既可以传入不同的model,又可以传入不同的属性。

 

  以上。

posted @ 2020-08-18 13:56  顾星河  阅读(654)  评论(0编辑  收藏  举报