1. 什么时候需要重写Object.equals方法

如果类具有自己特有的“逻辑相等”概念(不同于对象等同的概念),而且超类还没有覆盖equals以实现期望的行为,这时我们就需要覆盖equals方法。

这通常属于“值类(value class)”的情形。值类仅仅是一个表示值的类,例如Integer或Date。程序员在利用equals方法来比较值对象的引用时,希望知道它们在逻辑上是否相等,而不是想了解它们是否指向同一个对象。

有一种“值类”不需要覆盖equals方法,即用实例受控确保“每个值至多只有一个对象”的类,例如枚举类型。对于这样的类,逻辑相同与对象等同是一回事。

2.equals的通用约定

· 自反性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)就会一致地返回true,或者一致返回false;因此,equals方法里面不应该依赖任何不可靠的资源

· 对于任何非null的引用值x, x.equals(null)必须返回false

3.一些不好的示例

1)违反对称性。

定义如下一个类。

 class CaseInsensitiveString {

     private String insensitiveString;

     public CaseInsensitiveString(String string) {
this.insensitiveString = string;
} public String getString() {
return this.insensitiveString;
} @Override
public boolean equals(Object object) {
if (null == this.insensitiveString) {
return false;
} if (object instanceof CaseInsensitiveString) {
return this.insensitiveString
.equalsIgnoreCase(((CaseInsensitiveString) object)
.getString());
} if (object instanceof String) {
return this.insensitiveString.equals((String) object);
} return false;
}
}

该类的equals方法,首先违反了对称性。

 public class App {

     public static void main(String[] args) {
CaseInsensitiveString cis = new CaseInsensitiveString("whatever");
String string = "whatever";
System.out.println(cis.equals(string));
System.out.println(string.equals(cis));
} }

返回结果将是true和false。

2)考虑继承的场景。

假设下面有一个圆形的类,只要半径相同,就认为两个圆逻辑上等同。

 class Circle {
private int radius; public Circle(int radius) {
this.radius = radius;
} public int getRadius() {
return this.radius;
} @Override
public boolean equals(Object object) {
if (object instanceof Circle) {
return this.radius == ((Circle) object).getRadius();
} return false;
}
}

假如想要扩展这个类,为其加上一个颜色color的属性,那么其子类的equals方法要怎么写呢?

 class ColorCircle extends Circle {

     public static enum COLOR {
RED, WHITE, BLUE, GREEN, YELLOW
}; private COLOR color; public COLOR getColor() {
return this.color;
} public ColorCircle(int radius, COLOR color) {
super(radius);
this.color = color;
// TODO Auto-generated constructor stub
}
}

如果不覆盖equals方法,使用其父类的方法,那么,新添加的属性就会被忽略。虽然这样做不违反约定,但这样显然是无法接受的,否则为什么要新增一个属性?

那么,为了突出新增加的属性,提供如下的equals方法。

 @Override
public boolean equals(Object object) {
if (!(object instanceof ColorCircle)) {
return false;
} return super.equals(object)
&& ((ColorCircle) object).getColor() == this.color;
}

这样做,违反了对称性。下面实例化了两个对象,一个是以Circle,另一个是带颜色的ColorCircle,很明显,打印出来的结果一个是true,一个是false.

 public static void main(String[] args) {
Circle circle = new Circle(1);
ColorCircle colorCircle = new ColorCircle(1, COLOR.BLUE);
System.out.println(circle.equals(colorCircle));
System.out.println(colorCircle.equals(circle));
}

那么也许你会说,这种情况,那就判断其是子类还是父类,采用不同的策略进行比较。

 @Override
public boolean equals(Object object) {
if(!(object instanceof Circle)) {
return false;
} if (!(object instanceof ColorCircle)) {
return super.equals(object);
}
else {
return super.equals(object)
&& ((ColorCircle) object).getColor() == this.color;
}
}

这样是解决了对称性的问题了,但又带来了另一个问题,传递性。

     public static void main(String[] args) {
ColorCircle colorCircle = new ColorCircle(1, COLOR.BLUE);
Circle circle = new Circle(1);
ColorCircle anotherColorCircle = new ColorCircle(1, COLOR.RED);
System.out.println(colorCircle.equals(circle));
System.out.println(circle.equals(colorCircle));
System.out.println(colorCircle.equals(anotherColorCircle));
}

如上,colorCircle和circle等同,circle与anotherColorCircle等同,而很明显,colorCircle和anotherColorCircle是不等同的。

事实上,这是面向对象语言中,关于等价关系的一个基本问题。我们无法在扩展可实例化的类的同时,既增加新的值组件,同时又保留equals约定,除非愿意放弃面向对象的抽象带来的优势。

注意,这里说的是可实例化的类的扩展,对于抽象类不影响。

当然,你也可以用getClass测试代替instanceof测试,可以扩展可实例化的类和增加新的值组件,同时保留equals约定:

     @Override
public boolean equals(Object object) {
if (null == object || object.getClass() != getClass()) {
return false;
} ColorCircle colorCircle = (ColorCircle) object;
return this.getRadius() == colorCircle.getRadius()
&& this.color == colorCircle.getColor();
}

但是违背了里氏替换原则:任何基类出现的地方都可以无差别地使用子类替换。很明显,如果两个Circle等同,此时将其中一个替换为ColorCircle类,则不再等同。因此,子类应尽量少重写父类的方法。

总结一下,父类和子类的这种equals方法的重写问题的原因,首先,是因为父类是一个可实例化的类。那么就可能出现父类对象与子类对象比较的场景。所以对于抽象类来讲,并不影响,因为其无法实例化。

其次,是因为子类增加了一些标识身份的属性,必须在判断等同时使用。那么当父类与子类两个对象进行比较时,就会出现这种冲突,就需要思考,这个属性在比较的时候到底要不要使用。

因此,当【父类可实例化】且【子类新增属性必须参与到equals比较】的时候,equals方法总是会违反某些原则。

一个可以采用的方法是,组合优先于继承。在这里就不展开讨论了。

重写equals方法的约定的更多相关文章

  1. Effective Java 第三版——10. 重写equals方法时遵守通用约定

    Tips <Effective Java, Third Edition>一书英文版已经出版,这本书的第二版想必很多人都读过,号称Java四大名著之一,不过第二版2009年出版,到现在已经将 ...

  2. Java中Set的contains()方法——hashCode与equals方法的约定及重写原则

    转自:http://blog.csdn.net/renfufei/article/details/14163329 翻译人员: 铁锚 翻译时间: 2013年11月5日 原文链接: Java hashC ...

  3. Effective Java 第三版——11. 重写equals方法时同时也要重写hashcode方法

    Tips <Effective Java, Third Edition>一书英文版已经出版,这本书的第二版想必很多人都读过,号称Java四大名著之一,不过第二版2009年出版,到现在已经将 ...

  4. 重写equals方法(未完)

    equals方法是我们日常编程中很常见的方法,Object中对这个方法的解释如下: boolean equals(Object obj) 指示其他某个对象是否与此对象“相等”. 查看该方法的底层代码如 ...

  5. 为什么重写equals方法时,要求必须重写hashCode方法?

    1 equals方法 Object类中默认的实现方式是  :   return this == obj  .那就是说,只有this 和 obj引用同一个对象,才会返回true. 而我们往往需要用equ ...

  6. 为什么重写equals方法,还必须要重写hashcode方法

    一.equals方法和hashcode的关系 根据Object.hashCode的通用约定: 如果两个对象相同(equals方法返回true),那么hashcode也相等.(图1) 如果两个对象的ha ...

  7. java重写equals方法

    @Override public int hashCode() { return task.getId(); } @Override public boolean equals(Object obj) ...

  8. 重写equals()方法时,需要同时重写hashCode()方法

    package com.wangzhu.map; import java.util.HashMap; /** * hashCode方法的主要作用是为了配合基于散列的集合一起正常运行,<br/&g ...

  9. 为什么重写equals方法还要重写hashcode方法?

    我们都知道Java语言是完全面向对象的,在java中,所有的对象都是继承于Object类.Ojbect类中有两个方法equals.hashCode,这两个方法都是用来比较两个对象是否相等的. 在未重写 ...

随机推荐

  1. 浅谈 HTML5 的 DOM Storage 机制 (转)

    在开发 Web 应用时,开发者有时需要在本地存储数据.当前浏览器支持 cookie 存储,但其大小有 4KB 的限制.这对于一些 Ajax 应用来说是不够的.更多的存储空间需要浏览器本身或是插件的支持 ...

  2. 部署解决方案包 (SharePoint Server 2010)

    转:http://technet.microsoft.com/zh-cn/library/cc262995(v=office.14).aspx 本文介绍各个解决方案包,及其在 Microsoft Sh ...

  3. C# 根据Word模版生成Word文件

    指定的word模版 2,生成word类 添加com Microsoft word 11.0 Object Library 引用 using System; using System.Collectio ...

  4. 基于WebForm+EasyUI的业务管理系统形成之旅 -- 登录窗口(Ⅱ)

    上篇<基于WebForm+EasyUI的业务管理系统形成之旅 -- 系统设置>,主要是介绍系统浏览器在线下载安装,这些前期准备是非常重要的. 最近忙于将工程管理系统中各个模块,用业务流程方 ...

  5. C#实现Zip压缩解压实例【转】

    本文只列举一个压缩帮助类,使用的是有要添加一个dll引用ICSharpCode.SharpZipLib.dll[下载地址]. 另外说明一下的是,这个类压缩格式是ZIP的,所以文件的后缀写成 .zip. ...

  6. 浅谈PetShop之使用存储过程与PLSQL批量处理(附案例)

    1       大概思路 备注:黄色为影响参数 2       PetShop4的经典数据库连接代码回顾 PetShop4有3个函数,具体有: ExecuteReader:可以读一个表的记录,只能读不 ...

  7. LoadRunner 参数化详解

    可能稍微对LR 有些了解的人都知道LR 参数化的功能,今天请跟我来一起好好理一下. 首先,咱们先说一下为啥要用参数化?假如您录制了一个登录的脚本,如下所示, web_submit_form(" ...

  8. shell中常用的特殊符号

    在shell中常用的特殊符号罗列如下: # ; ;; . , / \ 'string'| ! $ ${} $? $$ $* "string"* ** ? : ^ $# $@ `co ...

  9. Storm系列(五)架构分析之Nimbus启动过程

    启动流程图   mk-assignments 功能:对当前集群中所有Topology进行新一轮的任务调度. 实现源码路径: \apache-storm-0.9.4\storm-core\src\clj ...

  10. hdu5601-N*M bulbs(黑白棋盘染色)

    一个矩形,一个人从左上角走到右下角,每走过一个位置把0变成1,1变成0. 求有没有可能他离开之后所有的数都是0 假设这个矩形是一个棋盘,黑白相间. 这样会发现从一个颜色走到相同颜色可以对棋盘不产生任何 ...