[Effective Java] (08)覆盖equals时请遵守通用规定

1. 不用覆盖的情况
  • 类的每个实例本质上都是唯一的。对于代表活动实体而不是值(value)的类来说确实如此,如Thread;
  • 不关心类是否提供了“逻辑相等(logical equality)”的测试功能。简单来说就是该类不需要这个功能;
  • 超类已经覆盖了equals,从超类继承过来的行为对于子类也是合适的。那我还费啥事重写equals;
  • 类是私有的或者包集私有的,可以确定它的equals方法永远不会被调用。在这种情况下应该覆盖equals方法的,为的是防止它被意外调用,代码如下:
@Override
public boolean equals(object o) {
    throw new AssertionError();          //Method is never called
}
2. 需要覆盖的情况

类具有自己特有的“逻辑相等”概念(不同于对象相等概念),而且超类还没有覆盖equals以实现期望的行为,这时,我们就需要覆盖equals方法。主要目的是知道其在逻辑上是否相等,而不是想了解它们是都指向同一个对象。

  • “每个值至多只存在一个对象”的“值类”不需要覆盖equals方法;还如枚举类型也属于这样的类。

equals的通用约定:

  • 自反性(reflecive)
  • 对称性(symmetric)
  • 传递性(transitive)
  • 一致性(consistent)
  • 对于任何非null的引用值x,x.equals(null)必须返回false

既可扩展不可实例类又增加值组件,采用复合优先于继承的方式(权宜之计)。不在让类B扩展类A,而是在类B中加入一个私有的A域。

3. 实现高质量equals方法诀窍★
  • 使用==操作符检查“参数是否为为这个对象的引用”;
  • 使用instanceof操作符检查“参数是否为正确的类型”;
  • 把参数转换成正确的类型;
  • 对于该类中的每个“关键(significant)”域,检查参数中的域是否与该对象中对应的域相匹配;
  • 考虑上述编写的equals方法,它是否是对称的、传递的、一致的?

示例代码:

public class PhoneNumber {
    private final short areaCode;
    private final short prefix;
    private final short lineNumber;
    
    @Override
    public boolean equals(Object obj) {
        
        if (obj == this) {
            return true;
        }
        if (!(obj instanceof PhoneNumber)) {
            return false;
        }
        PhoneNumber pn = (PhoneNumber)obj;
        return pn.lineNumber == lineNumber
            && pn.prefix == prefix
            && pn.areaCode == areaCode;
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 尽管Object是一个具体类,但是设计它主要是为了扩展。它所有的非final方法(equals、hashCode、...
    每天学点编程阅读 662评论 0 3
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,754评论 18 399
  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,356评论 11 349
  • 1. 尽量避免覆盖equals方法: 因为覆盖equals方法看似很简单,但实际上有许多覆盖方式会导致错误,并且后...
    大海孤了岛阅读 311评论 0 0
  • 我就生长在小路的旁边 经历风雨的考验 虽然只是一片微小的叶子 但我依然在冬季来临之际 迎接风雪的到来 喜欢积雪压在...
    秋之沭阳阅读 223评论 0 7