C#中判斷兩個對象是否相等有Equals、RefrenceEquals和==三種,其中==為運算子,其它兩個為方法,而Equals又有兩種版本,一個是靜態,一個是虛擬,虛擬可以被實體類重寫,靜態在方法體內也是調用虛擬,如下:
複製代碼 代碼如下:public static bool Equals(object objA, object objB)
{
return ((objA == objB) || (((objA != null) && (objB != null)) && objA.Equals(objB)));
}
public virtual bool Equals(object obj)
{
return InternalEquals(this, obj);
}
[MethodImpl(MethodImplOptions.InternalCall)]
internal static extern bool InternalEquals(object objA, object objB);
[ReliabilityContract(Consistency.WillNotCorruptState, Cer.Success)]
public static bool ReferenceEquals(object objA, object objB)
{
return (objA == objB);
}
以上為reflector反編譯的代碼。它們的區別與聯絡總結如下:
1. RefrenceEquals顧名思義,在任何情況下都是判斷兩個對象的引用是否相等,對於實值型別,因為每次判斷前都必須進行裝箱操作,也就是每次都產生了一個臨時的object,因而永遠返回false。String類型比較特殊,只要字元相同永遠是同一個引用,字元不同就是不同的引用,即使通過傳遞賦值如:string str1 = "a"; string str2 = str1; str2 = "b";這時str1,str2依舊是不同引用。
2. ==與Equals並無本質區別,它們大多數情況下都是一樣的,對於基本實值型別,判斷的是值是否相等,對於參考型別,判斷的則是引用是否一樣。值得注意的是,自訂的實值型別struct,本身並不支援運算子==,強行使用將會出現編譯錯誤。並且,鑒於Equals是虛方法,它可以被具體類重寫,因此需要具體問題具體分析。
3. 前面提過,靜態Equals本質上也是調用虛擬Equals,它們的區別在於調用時,虛擬要考慮對象是否為空白,否則會拋異常,而靜態則無需考慮。
以下為測試代碼: 複製代碼 代碼如下:class Program
{
static void Main(string[] args)
{
//AAA a1 = new AAA { Name = "a1", Age = 22 };
//AAA a2 = new AAA { Name = "a1", Age = 22 };
//int a1 = 123;
//int a2 = 123;
string a1 = "abc";
string a2 = "abc";
Console.WriteLine(string.Format("==: {0}", a1 == a2));
Console.WriteLine(string.Format("Equals: {0}", a1.Equals(a2)));
Console.WriteLine(string.Format("Static Equals: {0}", Object.Equals(a1, a2)));
Console.WriteLine(string.Format("ReferenceEquals: {0}", ReferenceEquals(a1, a2)));
Console.Read();
}
}
// Class or Struct
struct AAA
{
public string Name { get; set; }
public int Age { get; set; }
}