[C#] == VS Equals

声明:本篇博客翻译自:https://www.codeproject.com/Articles/1111680/equalsequals-VS-Equals-in-Csharp

由于水平(技术水平+英语理解能力)有限/不足,肯定会有所疏漏/错误,请及时指正。

介绍:

在C#中,可以使用“==”和一个可重载的"Equals()"方法来对变量进行比较。但是很多开发者并不知道两者之间的区别。在这篇博客中将讨论两者之间的区别和两者的使用场景。"=="和"Equals()"比较没有区别除了在做string的比较时。

比较规则:C#中的比较无非是对值类型和引用类型进行比较。当对值类型进行比较时,比较的是值类型的“内容”;当对引用类型进行比较时,比较的是对象的引用是否相同(内存地址)而不是“内容”。上述规则对"=="和"Equals()"均适用。

场景一:值类型比较

当对值类型(int, double, byte etc)进行比较时, "=="和"Equals()"都是基于值类型的内容比较。当值类型的内容相同,则两者相等。

Console.WriteLine("Value Type Compare...");
int i = 10;
int y = 10;
Console.WriteLine(i == y);        // true
Console.WriteLine(i.Equals(y)); // true

场景二:引用类型比较

当对引用类型(object, interface)进行比较时,比较的是引用(内存地址)是否相同,下面代码中obj1和obj2使用"=="和“Equals”进行比较,结果为:false.尽管两个对象的内容一样的。但是内容地址不一样。

class Customer
{
    public string Name { get; set; }
}

Customer obj1 = new Customer();
obj1.Name = "Jimmy-Yang";

Customer obj2 = new Customer();
obj2.Name = "Jimmy-Yang";

Console.WriteLine(obj1 == obj2);        // false
Console.WriteLine(obj1.Equals(obj2));   // false

如果obj1与obj2指向同一个引用(内存地址)。

Customer obj1 = new Customer();
obj1.Name = "Jimmy-Yang";

Customer obj2 = obj1;

Console.WriteLine(obj1 == obj2);        // true
Console.WriteLine(obj1.Equals(obj2));   // true

场景三:string比较

string类型是引用类型,那么下面代码中"str"创建一个string对象,并且在堆内存块上存储"test"字符。当创建str1后,虽然存储的内容一样,但是应该存储在不同的内存区域。

object str = "test";
object str1 = "test";
Console.WriteLine(str == str1);         // true
Console.WriteLine(str.Equals(str1));    // ture

执行结果发现str与str1是相等的。这是为什么呢?
注意:C#的string类型遵循一条内部的规则。如果str与str1的内容相同,则两者指向系统的的内存地址。因此“==”和"Equals"会返回true。

但是注意下面的代码,我们创建了一个单独对象,然后赋值给string类型。此时“==”比较将返回false,Equals()将返回true。这是两者一个不同的地方。

object str = new string(new char[] { 't', 'e', 's', 't' });
object str1 = new string(new char[] { 't', 'e', 's', 't' });
Console.WriteLine(str == str1);         // false
Console.WriteLine(str.Equals(str1));    // true

场景四:编译时VS运行时

两者之间在做类型检查时的区别。==在编译时做类型检查;Equals()在运行时做类型检查。请看下面的代码,在“==”会有一个警告信息。

场景五:NULL

当比较的对象为null时,“==”不会抛出异常,"Equals()"比较null对象时,会直接crash。

使用场景

"=="是C#提供的一个运算符,“Equals()”是基类提供的一个虚方法,可以在子类中继承,并编写自身比较逻辑。换句话说,“==”是一个C#的一个特性,而Equals是OOP的一个特性。两者的区别一个基于单纯的内容和引用(内存地址)比较。另一种是基于语义(Semantics, 此处我翻译成语义,其实更确切的应该是对象之间的含义)上的比较。例如:1 <> 10 (从技术层面上讲),但是现实生活中,1元=10角,此时1=10在含义上。

  • 如果单纯从技术层面上比较请使用“==”;
  • 如果从语义上比较(例如:1元=10角),你需要使用Equals方法并提供合适的比较逻辑。

总结

请看下面的表格:

    ==   Equals
Usage Technical based Semantic based
Value types Content based Comparison Content based Comparison
Objects Reference based Comparison Reference based Comparison
String  Content based Comparison Content based Comparison
String with no interning Reference based Comparison Content based Comparison
Type checking Compile time Run time
Nulls Works  Can crash
posted @ 2017-08-29 12:58  Yang-Fei  阅读(303)  评论(0编辑  收藏  举报