程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> .NET網頁編程 >> C# >> C#入門知識 >> 重載Object的虛方法——重載Equals和運算符

重載Object的虛方法——重載Equals和運算符

編輯:C#入門知識

Object是所有類的頂級父類,而Object又提供了四個虛方法:

Equals , GetHashCode, ToString, Finalize。

那麼在這個系列文章中,我們就看下,我們對這四個方法的利用。

首先是引用類型重載Equals,我分成三步:

1. 空值驗證

2. 類型驗證

3. 比較驗證

代碼如下:

class Person
{
    public string Name { get; set; }
    public int Age { get; set; }
    public City MyCity { get; set; }
    public override bool Equals(object obj)
    {
        if (obj == null) 
        { 
            return false; 
        }
        if (obj.GetType() != this.GetType()) 
        { 
            return false; 
        }
        Person personTemp = obj as Person;
        if (!Object.Equals(this.MyCity, personTemp.MyCity))
        {
            return false;
        }
        if (this.Age != personTemp.Age || this.Name != personTemp.Name)
        {
            return false;
        }
        return true;
    }
}

 

在此需要注意的是,在比較引用類型屬性的值是,需要使用Object的靜態方法去比較,主要是為了防止屬性值為null而拋出異常。我們來看下Object的靜態Equals實現就明白了:

public static bool Equals(object objA, object objB)
{
    return ((objA == objB) || (((objA != null) && (objB != null)) && objA.Equals(objB)));
}

 

呵呵,很漂亮的實現。解釋一下,其實就是首先比較兩者是否指向同一塊引用,然後判斷兩者是否都不為空,最後來調用類型的Equals重載方法。

接下來,我們看下,如果這個時候我們實現了一個Person類的子類,我們該怎麼寫?

class Programmer:Person
{
    public int CodeRowCount { get; set; }
    public override bool Equals(object obj)
    {
        if (!base.Equals(obj)) { return false; }
        Programmer pTemp = (Programmer)obj;
        if (pTemp.CodeRowCount != this.CodeRowCount) { return false; }
        return true;
    }
}

 

來簡單解釋一下,由於Person已經判斷了obj是否為空啊,類型是否相等,基類的字段是否相等,因此我們不需要再操心了,我們只需要比較子類獨有的字段是否相等即可。

這裡我們強調下,在Object默認的Equals實現中,比較的是兩個對象是否指向了同一個引用,因此,如果我們的父類沒有重載Equals方法,那麼我們的這個版本將永遠都是錯誤的,因此,我們也可以看出實現Equals方法的重要性吧,呵呵!

最後是值類型(主要是結構體)的重載Equals的方法,首先讓我們看看所有值類型的父類System.ValueType對於Equals的實現:

public override bool Equals(object obj)
{
    if (obj == null)
    {
        return false;
    }
    RuntimeType type = (RuntimeType)base.GetType();
    RuntimeType type2 = (RuntimeType)obj.GetType();
    if (type2 != type)
    {
        return false;
    }
    object a = this;
    if (CanCompareBits(this))
    {
        return FastEqualsCheck(a, obj);
    }
    FieldInfo[] fields = type.GetFields(BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance);
    for (int i = 0; i < fields.Length; i++)
    {
        object obj3 = ((RtFieldInfo)fields[i]).InternalGetValue(a, false);
        object obj4 = ((RtFieldInfo)fields[i]).InternalGetValue(obj, false);
        if (obj3 == null)
        {
            if (obj4 != null)
            {
                return false;
            }
        }
        else if (!obj3.Equals(obj4))
        {
            return false;
        }
    }
    return true;
}

 

方法很長,我來解釋一下:

首先,依然是來判斷obj是否為空;

接下來,來得到兩個對象的類型,在這裡出現了一個類是RuntimeType,我們Reflector下這個類:

image

是一個Internal類型,程序集外無法訪問,但是我們通過名稱和其中的屬性和方法名大概可以猜出,這是一個用於針對運行時反射而專門設計的類型。

接下來出現了CanCompareBitsFastEqualsCheck這兩個方法,在Reflector中無法看到實現,但是根據方法名,我猜想應該是判斷這個對象是否可以按位比較(我不是很理解,是指的序列化麼?),如果可以的話,直

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved