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下這個類:
是一個Internal類型,程序集外無法訪問,但是我們通過名稱和其中的屬性和方法名大概可以猜出,這是一個用於針對運行時反射而專門設計的類型。
接下來出現了CanCompareBits,FastEqualsCheck這兩個方法,在Reflector中無法看到實現,但是根據方法名,我猜想應該是判斷這個對象是否可以按位比較(我不是很理解,是指的序列化麼?),如果可以的話,直