一、equals方法
1.一、简介
equals()方法,定义在Object类中,源码以下:html
public boolean equals(Object obj) {
return (this == obj);
}
也就是说,在Object类中,比较的是两个对象的引用是否相同。
当咱们建立一个新的类而没有覆盖equals方法时,调用equals方法即比较两个对象的引用是否相同,此时与使用“==”比较是同样的。java
1.二、equals与==
参考:浅谈Java中的equals和==测试
String类对equals方法进行了重写,用来比较指向的字符串对象所存储的字符串是否相等。其余的一些类诸如Double,Date,Integer等,都对equals方法进行了重写用来比较指向的对象所存储的内容是否相等。flex
总结来讲:ui
- 对于==,若是做用于基本数据类型的变量,则直接比较其存储的 “值”是否相等;若是做用于引用类型的变量,则比较的是所指向的对象的地址
- 对于equals方法(注意:equals方法不能做用于基本数据类型的变量),若是没有对equals方法进行重写,则比较的是引用类型的变量所指向的对象的地址;诸如String、Date等类对equals方法进行了重写的话,比较的是所指向的对象的内容。
二、不须要覆盖equals方法
当有以下条件时,不须要覆盖equals方法:this
- 类的每一个实例本质上都是惟一的。
- 不关心类是否提供了“逻辑相等(logical equality)”的测试功能。
- 超类已经覆盖了equals,从超类继承过来的行为对于子类也是适合的。
- 类是私有的或是包级私有的,能够肯定它的equals方法永远不会被调用。
三、覆盖equals方法
3.一、何时该覆盖
若是类具备本身特有的“逻辑相等”概念(不一样于对象等同的概念),并且超类尚未覆盖equals以实现指望的行为,这时就须要覆盖equals方法。
这一般属于“值类(value class)”的情形。值类仅仅是一个表示值的类,例如Ingeter或Date,在利用equals方法来比较值对象的引用时,但愿知道它们在逻辑上是否相等,而不是想了解它们是否指向同一个对象。.net
3.二、覆盖时遵照约定
等价关系(equivalence relation):code
- 自反性(reflexive):对于任何非null的引用值x,x.equals(x)必须返回true。
- 对称性(symmetric):对于任何非null的引用值x和y,当且仅当y.equals(x)返回true时,x.equals(y)必须返回true。
- 传递性(transitive):对于任何非null的引用值x,y和z,若是x.equals(y)返回true,且y.equals(z)返回true,那么x.equals(z)也必须返回true。
- 一致性(consistent):对于任何非null的引用值x和y,只有equals的比较操做在对象中所用的信息没有被修改,屡次调用x.equals(y)总会一致地返回相同的结果。
- 非空性(Non-nullity):对于任何非null的引用值x,x.equals(null)必须返回false。
四、实现高质量equals
- 使用“==”操做符检查“参数是否为这个对象的引用”。
- 使用instanceof操做符检查“参数是否为正确的类型”。
- 把参数转换成正确的类型。
- 对于该类中的每一个“关键(significant)”域,检查参数中的域是否与该对象中对应的域相匹配。
- 当你编写完成了equals方法以后,应该问本身三个问题:它是否对称的、传递的、一致的?
五、告诫
- 覆盖equals时总要覆盖hashCode
- 不要企图让equals方法过于智能
- 不要将equals声明中的Object对象替换为其余的类型。
六、参考