第八条 在改写equals的时候请遵守通用约定

一般以下几种情况,不适宜覆盖equals方法

1.类的每个实例本质上都是唯一的,对于代表活动实体而不是值的类确实如此,例如Thread.

2.不关心类是否提供了“逻辑相等”的测试功能

3.超类已经覆盖了equals,从超类继承过来的行为对子类也是合适的

4.类是私有的或者包级私有的,可以确定它的equals方法永远不会被调用。

那什么情况应该覆盖Object.equals呢?如果类具有自己特有的“逻辑相等”的概念(不同于对象等同的概念),而且超类没有覆盖equals以实现期望的行为,就需要进行覆盖,这通常属于“值类”的情形,例如Integer和Date。当使用equals来比较对象, 是希望他们在逻辑上是否相等, 而不是指向同一对象, 或者用来作为Map的key以及集合Set中的元素时, 就必须复写equals方法.

实例受控,确保“每个值最多只存在一个对象”的类,枚举通常属于这种类型。对于枚举类型来说, 逻辑相等与对象相等是同一回事, 因此不需要覆盖equals方法。

equals的改写规范:

1)自反性:对于任何非null的引用值x,x.equals(x)一定为true

2)对称性:对于任何非null的引用值x和y,当且仅当x.dquals(y)为true;那么y.equals(x)也必须为true

3)传递性:对于任何非null的引用值x和y和z,如果x.equals(y)为true,y.equals(z);那么x.equals(x)也必须为true

4)一致性:对于任何非null的引用值x和y,如果用于equals比较的对象信息没有被修改的话,那么多次调用x.dquals(y)返回的值是一致的

5)对于非null引用值x,x.equals(null)一定返回false

接下来是逐一解析上面几个原则:

2)对称性

public final class CaseInsensitiveString {
    private final String s;

    public CaseInsensitiveString(String s) {
        if (s == null)
            throw new NullPointerException();
        this.s = s;
    }

    // Broken - violates symmetry!
    @Override
    public boolean equals(Object o) {
        if (o instanceof CaseInsensitiveString)
            return s.equalsIgnoreCase(((CaseInsensitiveString) o).s);
        if (o instanceof String) // One-way interoperability!
            return s.equalsIgnoreCase((String) o);
        return false;
    }

    // This version is correct.
    // @Override public boolean equals(Object o) {
    // return o instanceof CaseInsensitiveString &&
    // ((CaseInsensitiveString) o).s.equalsIgnoreCase(s);
    // }

    public static void main(String[] args) {
        CaseInsensitiveString cis = new CaseInsensitiveString("Polish");
        String s = "polish";
        System.out.println(cis.equals(s) + "  " + s.equals(cis));
    }
}

上面代码中的equals企图和String进行比较操作,假设我们有一个不区分大小的字符串和一个普通的字符串:

  CaseInsensitiveString cis = new CaseInsensitiveString("Test");
        String s = "test";

此时cis.equals(s)会返回true,CaseInsensitiveString 类中做了兼容大小写的处理,但是String 的equals方法是不知道要不区分大小写1的,所以s.equals(cis)会返回false,违反了自反性

假如你把CaseInsensitiveString 放到一个集合中

List<CaseInsensitiveString> list = new ArrayList<>();
        list.add(cis);
        list.contains(s);

list.contains(s)有可能返回true,也可能是false,甚至会抛出RumtimeException

为了解决这个问题,只要企图与String互操作的这段代码从equals去掉即可

@Override
    public boolean equals(Object o) {
        return o instanceof CaseInsensitiveString&&s.equalsIgnoreCase(((CaseInsensitiveString) o).s);

    }

3)传递性:

首先以一个简单不可变的二维整形Point类作为开始

最新文章

  1. 《编写可维护的JavaScript》——JavaScript编码规范(七)
  2. hdu 1394 Minimum Inversion Number(树状数组)
  3. c++学习之容器细枝末节(1)
  4. ImageLoader介绍2
  5. iOS block 声明时和定义时的不同格式
  6. [Android Pro] AndroidStudio导出jar包
  7. jython语言学习笔记
  8. 【leetcode❤python】237. Delete Node in a Linked List
  9. 2_JavaScript日期格式化
  10. (转载)delphi文件流
  11. 1136. Parliament(二叉树)
  12. Mongodb集群节点故障恢复场景分析
  13. UVA 10304 Optimal Binary Search Tree
  14. Java设计模式探讨之单例模式
  15. See you~(二维树状数组)
  16. 用纯c语言完成16位模式下的引导程序
  17. POJ1860-Currency Exchange-判正环
  18. STL之pair对组
  19. 2017-2018-2 20155314《网络对抗技术》Exp9 Web安全基础
  20. c#中base64编码解码

热门文章

  1. .Net HttpClient form-data格式请求
  2. MU puzzle
  3. bugfree登录后报错PHP Fatal error: Call-time pass-by-reference has been removed in
  4. R中去除为NA的行--转载
  5. codevs 1085 数字游戏 dp或者暴搜
  6. form表单提交数据的数据格式
  7. Maven 三种archetype说明
  8. atom的初次尝试,activate-power-mode 插件和做gif
  9. 15个Android通用流行框架大全
  10. 1059: [ZJOI2007]矩阵游戏 二分图匹配