那么为什么在重写equals方法的时候需要重写hashCode方法呢?
主要是Object.hashCode的通用约定:
a. 在java应用程序运行时,无论何时多次调用同一个对象时的hsahCode()方法,这个对象的hashCode()方法的返回值必须是相同的一个int值.
b. 如果两个对象equals()返回值为true,则他们的hashCode()也必须返回相同的int值.
c. 如果两个对象equals()返回值为false,则他们的hashCode()返回值也必须不同.
String s1 = new String("1111"); String s2 = new String("1111"); s1.hashCode(); s2.hashCode(); System.out.println(s1.equals(s2));
s1和s2对象所指的内存地址是不一样的,一个对象的hashcode是内存地址进行hash运算而得,如果string没有重写hashcode,那么s1和s2的hashcode 也有可能是不一样的。如果用Obejct的equals方法,那么比较的手机两个对象在堆内存的地址,那么结果会是是false,但是在业务系统中我们需要的是对象属性是否一致,所以重写了equals方法。string的底层数据结构是char数组,所以重写equals的逻辑就是两个对象的char数组循环比较值。那么equals重写了,但是如果hashcode没有重写那就违反了Object.hashCode的通用约定,所以hashCode必须也重写。可以从源码看出,两个值相等的String 的hashCode一定相等。
public boolean equals(Object anObject) { if (this == anObject) { return true; } if (anObject instanceof String) { String anotherString = (String)anObject; int n = value.length; if (n == anotherString.value.length) { char v1[] = value; char v2[] = anotherString.value; int i = 0; while (n-- != 0) { if (v1[i] != v2[i]) return false; i++; } return true; } } return false; }
public int hashCode() { int h = hash; if (h == 0 && value.length > 0) { char val[] = value; for (int i = 0; i < value.length; i++) { h = 31 * h + val[i]; } hash = h; } return h; }
参考 JavaBean关于为什么要重写hashCode()方法和equals()方法及如何重写_阿武刚巴得-CSDN博客