[改善Java代码]覆写equals方法时不要识别不出自己

本文揭示了在JavaBean中覆写equals方法时的一个常见陷阱,即忽略对象自身的情况,导致自反性原则违反。通过案例分析,详细解释了问题所在,并提供了简单有效的解决方案。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

建议45: 覆写equals方法时不要识别不出自己

我们在写一个JavaBean时,经常会覆写equals方法,其目的是根据业务规则判断两个对象是否相等,比如我们写一个Person类,然后根据姓名判断两个实例对象是否相同,这在DAO(Data Access Objects)层是经常用到的。具体操作是先从数据库中获得两个DTO(Data Transfer Object,数据传输对象),然后判断它们是否是相等的,代码如下:

 1 class Person{  
 2     private String name;  
 3 
 4     public Person(String _name){  
 5        name = _name;  
 6     }  
 7 
 8     @Override  
 9     public boolean equals(Object obj) {  
10          if(obj instanceof Person){  
11            Person p = (Person) obj;  
12            return name.equalsIgnoreCase(p.getName().trim());  
13          }  
14          return false;  
15     }
16 
17     public String getName() {
18         return name;
19     }
20 
21     public void setName(String name) {
22         this.name = name;
23     }  
24 } 

覆写的equals做了多个校验,考虑到从Web上传递过来的对象有可能输入了前后空格,所以用trim方法剪切一下,看看代码有没有问题,我们写一个main:

 1 public class Client {  
 2     public static void main(String[] args) {  
 3          Person p1 = new Person("张三");  
 4          Person p2 = new Person("张三 ");  
 5      
 6          List<Person> l =new ArrayList<Person>();  
 7          l.add(p1);  
 8          l.add(p2);  
 9          System.out.println("列表中是否包含张三:"+l.contains(p1));  
10          System.out.println("列表中是否包含张三 :"+l.contains(p2));  
11     }  
12 }  

上面的代码产生了两个Person对象(注意p2变量中的那个张三后面有一个空格),然后放到List中,最后判断List是否包含了这两个对象。看上去没有问题,应该打印出两个true才是,但是结果却是:

1 列表中是否包含张三:true
2 列表中是否包含张三 :false

刚刚放到list中的对象竟然说没有,这太让人失望了,原因何在呢?List类检查是否包含元素时是通过调用对象的equals方法来判断的,也就是说constains(p2)传递进去,会依次执行p2.equals(p1)、p2.equals(p2),只要有一个返回true,结果就是true,可惜的是比较结果都是false,那问题就出来了:难道p2.equals(p2)也为false不成?

还真说对了,p2.equals(p2)确实是false,看看我们的equals方法,它把第二个参数进行了剪切!也就是说比较的是如下等式:

"张三 ".equalsIgnoreCase("张三") 

注意前面的“张三 ”是有空格的,那这个结果肯定是false了,错误也就此产生了。这是一个想做好事却办成了“坏事”的典型案例,它违背了equals方法的自反性原则:对于任何非空引用x,x.equals(x)应该返回true。

问题知道了,解决也非常容易,只要把trim()去掉即可,注意解决的只是当前问题,该equals方法还存在其他问题。

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值