“==”和Equals区别
相信很多朋友在面对,对象判等时经常会犹豫是用“==”还是Equals呢?有时候发现两者得到的结果相同,但有时候有不同,
究竟在什么情况下"==" 会相等,什么情况下Equals会不相等,看完本篇相信你心里自有结论。
一 基本概念
值相等:表示比较的两个对象的数据成员按内存位分别相等,即两个对象类型相同,并且具有相等和相同的字段
引用相等:表示两个引用指向相同对象实例,也就是同一内存地址,(由引用相等,可以推出其值相等,反之则不能)
二 不同类型"=="与Equals 判等结果
2.1 值类型
"=="运算符:表示值是否相等
"Equals":还是表示值是否相等 ,反编译后可以看到Equals 方法还是用了"=="运算符判等
public bool Equals(int obj)
{
return (this == obj);
}
所以对于值类型 不管是用 "=="运算符还是Equals方法得到的结果都是一样的(方法或运算符没有被重载)
2.2 引用类型
2.2.1 一般引用类型的 "=="运算符:表示引用地址是否相等,
两个对象虽然值相等,但是属于不同实例,所有引用地址也不一样
object a = new string(new char['a']);
object b = new string(new char['a']);
Console.WriteLine(a == b);//false
2.2.2一般引用类型的 Equals方法:表示值是否相等
object a = new string(new char['a']);
object b = new string(new char['a']);
Console.WriteLine(a.Equals(b));//true
2.2.3 特殊引用类型string: “==”运算符:表示值是否相等
string a = "123";
string b = "123";
Console.WriteLine(a==b);//true
对于为什么string 会比较值相等,相信大家都有疑问那请看下边的代码

注意:两个stirng对象在用"=="运算符判等时会调用 op_Equality()方法,这个方法里究竟做了什么操作呢,反编译一下string类 得到如下代码
我发现op_Equality()方法里居然重写了"=="运算符 且在其中调用了Equals方法,那string类的Equals方法又进行了哪些操作呢,请看图2


跟进Equals方法后发现居然是用"=="操作符判等了,瞬间我明白了大至如下:
因为string类以“==”操作符来处理值相等,
而重写“==”操作符的方法中又调用了Equals方法,
而Equals方法最终还是用"=="操作符判等,
我去~~带我们绕了个圈,最终发现string 类的 Equals方法与"=="操作符,得到结果是一样的
//特殊引用类型string
string a = "123";
string b = "123";
Console.WriteLine(a==b);//true
Console.WriteLine(a.Equals(b));//true
三 自定义引用类型 “==”与 Equal判等
3.1 "=="操作符:表示引用是否相等
因为两个实例处于托管堆中不同的块中所引用地址必然不同,返回false
class MyClass
{
public string str = "123";
}
MyClass myClass = new MyClass();
MyClass newClass = new MyClass();
Console.WriteLine(myClass == newClass);//false
3.2 Equals() 方法:表示相个实例是否相等
注意:为什么会是false 明明两个对象是一样的啊???为什么??为什么??? 想知道原因?请接着看
class MyClass
{
public string str = "123";
}
MyClass myClass = new MyClass();
MyClass newClass = new MyClass();
Console.WriteLine(myClass.Equals(newClass));//false
解释:所有类默认都继承了object类,Object类本身是带有几个方法 的,Equals就是其中之一,因为MyClass没有重载基类Object中的 Equals虚方法所以调用的是Object中的Equals方法 ,那Object中的Equals方法究竟又有什么操作呢?
反编译一下就知道鸟
[TargetedPatchingOptOut("Performance critical to inline across NGen image boundaries"), __DynamicallyInvokable]
public virtual bool Equals(object obj)
{
return RuntimeHelpers.Equals(this, obj);
}
RuntimeHelpers.Equals()方法官方说明:此方法判断指定的 Object 实例是否被视为相等
很明显 代码中 myClass 与 newClass 属于不同实例,所以会返回false
若要返回True则需要重载Object中的Equals方法如下
class OverrideClass
{
public string Str = "123";
//重载Object中Equals方法
public override bool Equals(object obj)
{
OverrideClass nclass = obj as OverrideClass;
//as 如果转换失败会返回null
if (nclass == null)
{
return false;
}
//对比对象属性是否相等
if (this.Str == nclass.Str)
{
return true;
}
else
{
return false;
}
}
}
OverrideClass oClass = new OverrideClass();
OverrideClass nClass = new OverrideClass();
Console.WriteLine(oClass.Equals(nClass));//true
四 总结
到这里“==”与Equals就写完了总结如下:
值类型:“==”操作符与Equals方法本质上没有区别都将得到相同结果
引用类型:"=="操作符表示引用地址是否相等,Equals方法表示 值 是否相等
自定义引用类型:"=="操作符表示引用地址是否相等,Equals方法 根据重载的情况得到不同结果
特殊引用类型string:"=="与Equals 存在相互调用,得到结果都一样
作者:RichardCui
出处:https://www.cnblogs.com/yachao1120/
本文版权归作者和博客园共有,欢迎转载,但未经作者同意必须保留此段声明,且在文章页面明显位置给出原文连接,否则保留追究法律责任的权利。该文章也同时发布在我的独立博客中-RichardCuiBlog。


浙公网安备 33010602011771号