1700442250
1700442251
更玄的是p1与e1、e2相等,但e1竟然与e2不相等,似乎一个简单的等号传递都不能实现。这才是我们要分析的真正重点:e1.equals(e2)调用的是子类Employee的equals方法,不仅仅要判断姓名相同,还要判断工号是否相同,两者工号是不同的,不相等也是自然的了。等式不传递是因为违反了equals的传递性原则,传递性原则是指对于实例对象x、y、z来说,如果x.equals(y)返回true, y.equals(z)返回true,那么x.equals(z)也应该返回true。
1700442252
1700442253
这种情况发生的关键是父类使用了instanceof关键字,它是用来判断是否是一个类的实例对象的,这很容易让子类“钻空子”。想要解决也很简单,使用getClass来代替instanceof进行类型判断,Person类的equals方法修改后如下所示:
1700442254
1700442255
public boolean equals(Object obj){
1700442256
1700442257
if(obj!=null&&obj.getClass()==this.getClass()){
1700442258
1700442259
Person p=(Person)obj;
1700442260
1700442261
if(p.getName()==null||name==null){
1700442262
1700442263
return false;
1700442264
1700442265
}else{
1700442266
1700442267
return name.equalsIgnoreCase(p.getName());
1700442268
1700442269
}
1700442270
1700442271
}
1700442272
1700442273
return false;
1700442274
1700442275
}
1700442276
1700442277
当然,考虑到Employee也有可能被继承,也需要把它的instanceof修改为getClass。总之,在覆写equals时建议使用getClass进行类型判断,而不要使用instanceof。
1700442278
1700442279
1700442280
1700442281
1700442283
编写高质量代码:改善Java程序的151个建议 建议48:覆写equals方法必须覆写hashCode方法
1700442284
1700442285
覆写equals方法必须覆写hashCode方法,这条规则基本上每个Javaer都知道,这也是JDK API上反复说明的,不过为什么要这样做呢?这两个方法之间有什么关系呢?本建议就来解释该问题,我们先来看如下代码:
1700442286
1700442287
public static void main(String[]args){
1700442288
1700442289
//Person类的实例作为Map的key
1700442290
1700442291
Map<Person, Object>map=new HashMap<Person, Object>(){
1700442292
1700442293
{
1700442294
1700442295
put(new Person(“张三”),new Object());
1700442296
1700442297
}
1700442298
1700442299
};
[
上一页 ]
[ :1.70044225e+09 ]
[
下一页 ]