所有对象共有的方法(第3章)

这是Joshua Blochs的《 有效的Java》第3章的简短摘要。我仅包括与自己相关的项目。

一般

等值合约将等价关系描述为:

  • x.equals(null) == false
  • 自反x.equals(x) == true
  • 对称 –如果x.equals(y) == truey.equals(x) == true
  • 可传递的 –如果x.equals(y) == truey.equals(z) == truex.equals(z) == true
  • 一致 –在相同的未修改对象上多次调用equals,返回相同的值。

有一些值得注意的属性可以改变:

  • 如果违反了以上约定,则其他对象(如List.contains() )的行为未定义。
  • 注意,如果基类是抽象的,那就很好了。
  • java.net.URL依赖于与URL关联的主机的IP地址,这些主机需要网络访问,因此会破坏一致性。
  1. 检查参数==this
  2. 使用的instance of检查类型是否正确
  3. 转换为正确的类型
  4. 比较重要领域

当您覆盖

哈希码由基于哈希的结构使用。 哈希码协定的最重要部分规定,相等的对象必须返回相等的哈希码。 此外,hashcode函数为不相等的对象返回不同的值,以提高性能。 如果没有正确的哈希码实现,则将相等的对象视为不相等的基于哈希的结构将表现不佳,甚至更糟。 如果将常量值作为hashCode提供,例如... return 42 ,则哈希表会退化为链接列表,并且程序应以线性时间运行,并以二次时间运行。

始终覆盖

…因为它使调试容易得多。

注意

实现Clonable使Object.clone()返回一个逐字段的副本,否则抛出CloneNotSupportedException 。 通常,克隆会创建一个对象,但会绕过构造函数。 实现clone存在若干挑战:

  • 通常,尤其是在扩展类时,在覆盖clone ,应返回super.clone()返回的对象以获取正确的类型。 这不是强制性的,它由用户决定,但是如果没有它,克隆可能会中断。
  • clone不会复制可变的对象字段,因此super.clone()将引用相同的对象字段。 必须手动克隆字段。
  • 从本质上讲,这意味着与克隆一起使用时,字段不能为“ final”,除非可以共享相同的字段值。
  • 由于clone在不使用构造函数的情况下创建对象,因此必须确保创建后所有不变式都是正确的。
  • 必须在内部列表/数组上递归调用clone

一般建议是避免使用和实现Object.clone() ,而是使用复制构造函数public Yum(Yum yum)或工厂,除非复制数组时。

实施

可比较处理订单比较,并且在使用例如TreeSetTreeMapsearchsort时是必需的。

  • 可比对象具有相似的合同,即equals ,如果被破坏,可能会导致行为不稳定。 合同要求对称,自反和传递。
  • equals是不符合compareTo可以创建一些集合重复。
  • Float和Double具有它们自己的静态compareTo方法,这些方法应简化处理浮点问题。
  • 减去整数以创建compareTo返回值时要小心,因为它可能会导致溢出(即在Integer.MAX_VALUE之外)并创建错误的返回值! 如果i为正值,而j为负值,则ij溢出并返回负值。

翻译自: https://www.javacodegeeks.com/2017/05/effective-java-method-common-objects.html