作为在Object中的equals方法和hashCode方法,或多或少我们在子类中都有重写过这两个方法,那么我们在重写这两个方法时需要注意些什么?就让我们通过这篇文章来聊一聊。
什么时候覆盖equals方法我就不再说了,相信熟悉Java的读者肯定都知道,我们着重聊一下如何书写规范的equls方法。因为一个不规范的equals方法将会造成集合无法表现出预期的行为。
重写equals方法需要保证equals满足以下特性:
该要求说明对象必须等于自身,如果违背这一条你会发下集合的contains方法无法告诉你正确的结果。
对称性要求两个对象对于它们是否相等要保持一致,下面这段代码违反了对称性
public class IgnoreCaseString {
private final String s;
public IgnoreCaseString(String s) { this.s = s; }
@Override public boolean equals(Object obj)
{ if (obj instanceof IgnoreCaseString)
{ return s.equalsIgnoreCase(((IgnoreCaseString) obj).s); }
if (obj instanceof String) { return s.equalsIgnoreCase((String) obj); } return false; }
public static void main(String[] args)
{ IgnoreCaseString ignoreCaseString = new IgnoreCaseString("Phone");
String string = "phone";
System.out.println(ignoreCaseString.equals(string));
System.out.println(string.equals(ignoreCaseString));
System.out.println("------------------------------");
List<String> list = new ArrayList<>();
list.add(string);
System.out.println(list.contains(ignoreCaseString));
}
}
这种情况通常发生在具有父子关系的对象中,子类增加的信息会影响到equals的比较结果。解决这种问题通常有两种方式,一种是通过getClass()的方式(具体的大家可以通过阅读Effective Java这本书),还有一种是在我们扩展类的功能时尽量使用复合而并不是使用继承,通过复合组件里面的域的比较也可以解决。
一致性要求如果两个对象相等,那么他们就必须保持相等,除非它们中有对象被修改了。
非空性要求所有的对象不等于null
在每个覆盖了equals方法的类中,必须重写hashCode方法。如果不这样做,会导致该类无法与所有基于散列的集合一起正常运作。hashCode方法需要遵循以下规定:
使用非0初始值的原因是让域初始值为0的那些域可以影响到散列值,如果使用0那么散列值不会再受这些域的影响,从而增加Hash冲突的可能性,从而降低了散列表的性能
31是一个奇素数,在Java中如果两个比较大的数相乘则会发生移除,31并不算一个比较大的数,其次选用31的理由是可以用移位和减法来代替乘法,31 * i = (i << 5) - i,现在的VM都可以自动完成这种优化,因此可以获得很好的性能提升
如果一个类是不可变的并且计算散列码的成本比较大,可以考虑把散列码缓存在对象内部,而不是每次请求时都重新计算(这一点在Kafka中也有所应用)。