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; }
}