关闭

建议11: 区别对待==和Equals

标签: Equals比较
210人阅读 评论(0) 收藏 举报
分类:

建议11: 区别对待==和Equals

在开始本建议之前,首先要明确概念“相等性”。CLR中将“相等性”分为两类:“值相等性”和“引用相等性”。如果用来比较的两个变量所包含的数值相等,那么将其定义为“值相等性”;如果比较的两个变量引用的是内存中的同一个对象,那么将其定义为“引用相等性”。

无论是操作符“==”还是方法“Equals”,都倾向于表达这样一个原则:

对于值类型,如果类型的值相等,就应该返回True。

对于引用类型,如果类型指向同一个对象,则返回True。

下面的代码输出所遵循的就是以上原则:

  1. static void ValueTypeOPEquals()  
  2. {  
  3.     int i = 1;  
  4.     int j = 1;  
  5.     //True  
  6.     Console.WriteLine(i == j);  
  7.     j = i;  
  8.     //True  
  9.     Console.WriteLine(i == j);  
  10. }  
  11.  
  12. static void ReferenceTypeOPEquals()  
  13. {  
  14.     object a = 1;  
  15.     object b = 1;  
  16.     //False  
  17.     Console.WriteLine(a == b);  
  18.     b = a;  
  19.     //True  
  20.     Console.WriteLine(a == b);  
  21. }  
  22.  
  23. static void ValueTypeEquals()  
  24. {  
  25.     int i = 1;  
  26.     int j = 1;  
  27.     //True  
  28.     Console.WriteLine(i.Equals(j));  
  29.     j = i;  
  30.     //True  
  31.     Console.WriteLine(i.Equals(j));  
  32. }  
  33.  
  34.  static void ReferenceTypeEquals()  
  35. {  
  36.     object a = new Person("NB123");  
  37.     object b = new Person("NB123");  
  38.     //False  
  39.     Console.WriteLine(a.Equals(b));  
  40.     b = a;  
  41.     //True  
  42.     Console.WriteLine(a.Equals(b));  

但是,我们同时也要了解,无论是操作符“==”还是“Equals”方法都是可以被重载的。比如,对于string这样一个特殊的引用类型,微软觉得它的现实意义更接近于值类型,所以,在FCL中,string的比较被重载为针对“类型的值”的比较,而不是针对“引用本身”的比较。

从设计上来说,很多自定义的类型(尤其是自定义的引用类型)会存在和string类型比较接近的情况。如例子中所举的类型Person,在现实生活中,如果两者的IDCode是相等的,我们就认为两者是同一个人,这个时候,就要重载Equals这个方法,代码如下所示:

  1. class Person  
  2. {  
  3.     public string IDCode { get; private set; }  
  4.  
  5.     public Person(string idCode)  
  6.     {  
  7.         this.IDCode = idCode;  
  8.     }  
  9.  
  10.     public override bool Equals(object obj)  
  11.     {  
  12.         return IDCode == (obj as Person).IDCode;  
  13.     }  

这时,再通过Equals去比较两个具有相同IDCode的Person对象的值,返回的就会是true,代码如下所示:

  1. object a = new Person("NB123");  
  2. object b = new Person("NB123");  
  3. //False  
  4. Console.WriteLine(a == b);  
  5. // True  
  6. Console.WriteLine(a.Equals(b)); 

这里,再引出操作符“==”和“Equals”方法之间的一点区别。一般来说,对于引用类型,我们要定义“值相等性”,应该仅仅去重载Equals方法,同时让“==”表示“引用相等性”。

注意 由于操作符“==”和“Equals”方法从语法实现上来说,都可以被重载为表示“值相等性”和“引用相等性”。所以,为了明确有一种方法肯定比较的是“引用相等性”,FCL中提供了Object. ReferenceEquals方法。该方法比较的是:两个示例是否是同一个示例。

转自:《编写高质量代码改善C#程序的157个建议》陆敏技

0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:39835次
    • 积分:722
    • 等级:
    • 排名:千里之外
    • 原创:9篇
    • 转载:157篇
    • 译文:0篇
    • 评论:0条
    文章分类