• C#中的ReferenceEquals、Equals以及==


        C#中有一共有四种相等性判断方法:

    //Object中定义的三个方法
    public static bool ReferenceEquals(object objLeft, object objRight);
    public static bool Equals(object objLeft, object objRight);
    public virtual bool Equals(object obj);
    
    //双等号
    public static bool operator == (MyClass left, MyClass right)

       

        ①Object.ReferenceEquals(left, right)静态方法:从名称中便可知它用来比较两者是否是相同的引用,我们也永远不应该去重写该方法。它对于值类型对象的比较永远返回false;对于两个null的比较永远返回true。

        ②Object.Equals(left, right)静态方法:该方法也永远不需要重写,因为它最终会把判断权交给参数left的实例Equls方法(代码如下),因此没有必要重写该方法,只需要保证实例Equals返回想要的结果即可:

    public static bool Equals(object left, object right)
    {
        if(object.ReferenceEquals(left,right)) //如果引用相同,则必定相等(包含两个都是null的情况)
        {
            return true;
        }
        if(object.ReferenceEquals(left ,null)|| object.ReferenceEquals(null,right)) //若只有一个是null,则必定不等
        {
            return false;
        }
        return left.Equals(right);//两个都不是null,且也不是同一个引用,则根据left的实例Equals方法来判断。
    }

        ③Object中的实例方法Equals,因为它是虚方法,所以可以在其他类中重写它。该方法的默认实现还是比较两者是否为同一个引用,即相当于ReferenceEquals。但是微软在所有值类型的基类System.ValueType中重写了该方法,用来比较值相等。

    • 在值类型中,我们仍然可能要重写该方法,以提高性能(默认方法的性能不高),另外如果我们的struct中包含了引用类型的成员,则应该重写该方法。
    • 在引用类型中,如果类的实例要作为字典的键,则应该重写Equals方法,使之比较值,而不是引用。

        重写Equals应该遵循的原则:自反性、对称性、传递性。即:a=a;若a=b,则b=a;若a=b,b=c,则a=c;另外两个对象要么相等要不不等,所以该方法不应该抛出异常。下面是重写模板:

    public class MyClass:IEquatable<MyClass>
    {
        //重写Object中的Equals方法
        public override bool Equals(object obj)
        {
            if (object.ReferenceEquals(obj, null)) //首先判断obj不能为空,否则后面对obj调用任何方法都将报错
            {
                return false;
            }
            if(object.ReferenceEquals(this,obj)) //在C#中this永远不会为空
            {
                return true;
            }
            if(this.GetType() != obj.GetType())
            {
                return false;
            }
            return this.Equals(obj as MyClass);
        }
        //实现IEquatable<T>中的Equals方法
        public bool Equals(MyClass other)
        {
            //省略
            return true;
        }
    }

        ④比较运算符==:对于引用类型,默认是比较引用的(System.String除外),对于值类型默认比较值,对于自定义的结构,如果不显示重载operator ==方法,则无法使用==。

    • 由于在C#中要求比较运算符必须成对重载,重载==运算符的同时也必须重载!=运算符,否则也会产生编译错误。
    • 如果要重载的话,运算符"=="、"!="  与  Equals方法、GetHashCode方法应该同时被重载,因为他们应该保持同样的相等逻辑。但不要再==中调用Equals,最好是在Equals中调用==。
  • 相关阅读:
    入职事业单位一年,真真的发现与私企大不同
    就算收入腰斩,我也决定离开私企,去吃公家饭了
    写了8年的代码,做过的项目都下线了,程序员的意义在哪里!
    程序员又双叕和产品打起来了,说说与产品经理的那些事。
    创业公司的技术总监,去上市公司面试,结果凉了。
    看了程序员的未来在哪里,有感
    有个程序猿要去当CEO了:(一)事情始末
    企业微信/政务微信 previewFile方法在有iframe情况下且iOS点击无效解决方案
    Mark: 实现个toy版的脚手架(RPC)
    搞技术的思维
  • 原文地址:https://www.cnblogs.com/ArtofDesign/p/3615212.html
Copyright © 2020-2023  润新知