有时候你想这么做:
Map<User, Profile> map = new HashMap<>();
// 添加一些数据
map.put(new User(1), getProfile(1));
// 取用这些数据
if (map.containsKey(new User(1)) {
doSomething();
}
但默认情况下这是不能成功的,因为在 HashMap 的实现中,是这么处理的:
// 代码有精简,改变了对 null 值的判断逻辑,不过这不是重点
if (key != null
&& e.hash == key.hashCode()
&& (e.key == key || key.equals(e.key)) {
return e;
}
注意,hashCode 相同,不一定 equals() 返回 true。
也就是说,我们要手动实现 equals()
和 hashCode()
才能达到我们的目的。
class User {
private Integer id;
@Override
public int hashCode() {
return this.id != null ? this.id : 0;
}
@Override
public boolean equals(Object obj) {
return obj instanceof User && (this.id.equals(((User) obj).id));
}
}
大功告成。
Think in Java 中设计 equals()
的五条原则
- 自反性。
即x.equals(x)
为 true - 对称性。
若x.equals(y)
为 true,那么y.equals(x)
为 true - 传递性。
若x.equals(y)
为 true 且y.equals(z)
为 true,那么x.equals(z)
也为 true - 一致性。
对于x.equals(y)
,只要判定信息不变,无论比较多少次结果均应相同。 - 若
x != null
为 true,那么x.equals(null)
为 false
网友评论