【发布时间】:2013-08-24 05:01:12
【问题描述】:
我知道每当equals 方法在Java 中被覆盖时,都需要覆盖哈希码。那只是一份合同。我试图理解这背后的逻辑。我正在阅读Joshua Bloch 的*Effective Java,我遇到了这段代码(第 9 项,第 45 页):
import java.util.HashMap;
import java.util.Map;
public final class PhoneNumber {
private final short areaCode;
private final short prefix;
private final short lineNumber;
public PhoneNumber(int areaCode, int prefix, int lineNumber) {
rangeCheck(areaCode, 999, "area code");
rangeCheck(prefix, 999, "prefix");
rangeCheck(lineNumber, 9999, "line number");
this.areaCode = (short) areaCode;
this.prefix = (short) prefix;
this.lineNumber = (short) lineNumber;
}
private static void rangeCheck(int arg, int max, String name) {
if (arg < 0 || arg > max)
throw new IllegalArgumentException(name + ": " + arg);
}
@Override
public boolean equals(Object o) {
if (o == this)
return true;
if (!(o instanceof PhoneNumber))
return false;
PhoneNumber pn = (PhoneNumber) o;
return pn.lineNumber == lineNumber && pn.prefix == prefix
&& pn.areaCode == areaCode;
}
// Broken - no hashCode method!
// A decent hashCode method - Page 48
// @Override public int hashCode() {
// int result = 17;
// result = 31 * result + areaCode;
// result = 31 * result + prefix;
// result = 31 * result + lineNumber;
// return result;
// }
// Lazily initialized, cached hashCode - Page 49
// private volatile int hashCode; // (See Item 71)
//
// @Override public int hashCode() {
// int result = hashCode;
// if (result == 0) {
// result = 17;
// result = 31 * result + areaCode;
// result = 31 * result + prefix;
// result = 31 * result + lineNumber;
// hashCode = result;
// }
// return result;
// }
public static void main(String[] args) {
Map<PhoneNumber, String> m = new HashMap<PhoneNumber, String>();
m.put(new PhoneNumber(707, 867, 5309), "Jenny");
System.out.println(m.get(new PhoneNumber(707, 867, 5309)));
}
}
这是他在文中提到的,我很难理解。
此时,您可能希望
m.get(new PhoneNumber(707, 867, 5309))返回“Jenny”,但它返回 null。注意两个 涉及到 PhoneNumber 实例:一个用于插入 HashMap 和第二个相等的实例用于(尝试) 恢复。 PhoneNumber 类未能覆盖 hashCode 导致 两个相等的实例具有不相等的哈希码,违反 哈希码合约。因此 get 方法可能会寻找 与它所在的哈希桶不同的哈希桶中的电话号码 由 put 方法存储
我不明白他所说的两个 PhoneNumber 实例是什么。我在m.put(new PhoneNumber(707, 867, 5309), "Jenny") 中创建了唯一的实例。我也再次寻找这个对象,它应该返回相同的哈希码,即使它从对象类继承了 hashCode 方法。
为什么会这样?这里的一些解释会很有帮助。
【问题讨论】:
-
以一种非常非技术性的方式思考它;您可以将哈希码视为相等的(快速)提示,而 equals 方法(缓慢)解决问题。如果两者不同步,那么显然可能会发生疯狂的事情
标签: java object equals hashcode effective-java