第8条:覆盖equals时请遵守通用约定
什么时候应该覆盖Object.equals呢?
如果类具有自己特定的“逻辑相等”的概念(不同于对象等同的概念),同时超类还没有覆盖equals以实现期望的行为,这时我们需要覆盖equals方法。
覆盖equals方法要遵守的通用约定,equals方法实现了等价关系:
自反性:对于非null值的x,x.equals(x);
对称性:对于非null值的x,y;当且仅当x.equals(y)时,y.equals(x)返回true;
传递性:对于非null值的x,y,z;如果x.equals(y)为true,y.equals(z)为true,那么x.equals(z)为true;
一致性:对于非null值的x,y,只要equals方法中的所用信息没有修改,多次调用equals方法返回的值一致。
对于非 null值,x.equals(null)返回false。
实现高质量equals方法的诀窍:
1.使用==操作符检查“参数是否为这个对象的引用”。
2.使用instanceof操作符检查“参数是否为正确的类型”。(过滤空值)
3.把参数转换成正确的类型。
4.对于该类中的每个关键域,检查参数中的域是否和该对象中的对应的域匹配。
对于float要使用Float.compare,double要使用Double.compare。
5.当你编写完equals方法的时候,看它是否满足等价关系。
第9条:覆盖equals方法时总要覆盖hashCode
相等的对象必须具有相同的hashcode。
为不相等的对象产生不相等的散列码。
必须排除equals比较计算中没有用到的任何域。
如果一个对象是不可变对象,应该将散列值缓存下来。
第10条:始终要覆盖toString
toString应该返回对象中包含的所有值得关注的信息
第11条:谨慎地覆盖clone
clone方法:无需调用构造器就可以创建对象
clone的一般含义,对于任何对象x,
x.clone() != x 为true
x.clone().getClass() == x.getClass() 为true
x.clone().equals(x) 为true
如果对象中包含的域引用了可变的对象,简单的clone实现会导致灾难性的后果。
因为简单的clone方法对可变的对象域,实行的是“浅拷贝”。
也就是拷贝的对象和原本的对象同时引用了一个可变对象域。
HashMap的clone方法就是浅拷贝(Shallow Copy)
所有实现了Cloneable接口的类都应该用一个共有的方法覆盖clone。此公有的方法首先调用super.clone方法,然后修正任何需要修正的域。一般情况下,这意味着要拷贝任何包含内部“深层结构”的可变对象,并用指向新对象的引用代替原来指向这些对象的引用。
如果该类只包含基本类型的域,或者指向不可变对象的引用,那就没有域需要修正。
对于不可变类,对象拷贝没有太大的意义。
另一种实现对象拷贝的好方法就是提供一个拷贝构造器或者拷贝工厂
public Yum(Yum yum)
public static Yum newInstance(Yum yum)
所有通用集合实现都提供了一个拷贝构造器。
第12条:考虑实现Comparable接口
如果类实现了Comparable接口就表明它的实例具有内在的排序关系。