Java equals方法实现总结
作者: Denlee
原文链接: http://blog.csdn.net/denlee/archive/2008/10/15/3081722.aspx
所有的对象都有标识(内存中的地址)和状态(对象的数据)。'=='运算符比较两个对象的地址,Object类的equals方法的默认实现也是按照内存地址比较对象是否相等,因此如果 object1.equals(object2)为true,表明object1变量和object2变量实际上引用同一个对象。
有些时候,默认的equals方法的实现就可以满足要求;但是很多时候我们需要比较对象的数据,特别是由数据库记录映射的对象。
Java规范建议equals方法遵循以下几个特性:
1) 自反性:对于任何非空引用 x, x.equals(Object) 将返回 true;
2) 对称性:对于任何引用 x 和 y,当且仅当 y.equals(x)返回 true 时,x.equals(y)返回 true;
3) 传递性:对于任何引用 x、y 和 z,如果x.equals(y) 返回true 并且 y.equals(z)也返回true,那么 x.equals(z) 也应该返回 true;
4) 一致性:如果 x 和 y 引用的对象没有改变,那么 x.equals(y)的重复调用应该返回同一结果;
5) 对任何非空引用 x, x.equals(null)应该返回false。
在实现equals方法时,应该根据实例域的类型进行不同的比较:
1) 对象域,使用equals方法
2) 类型安全的枚举,使用equals或==
3) 可能为null的对象域 : 使用 == 和 equals
4) 数组域 : 使用 Arrays.equals
5) 除float和double外的原始数据类型 : 使用 ==
6) float类型: 使用Float.foatToIntBits转换成int类型,然后使用==
7) double类型: 使用Double.doubleToLongBit转换成long类型,然后使用==
举例:
- //equals方法的参数保持为Object类型,否则在动态绑定调用机制会调用超类的equals方法
- public boolean equals(Object o) {
- //先检查是否是自比较
- if ( this == o ) return true;
- if ( o == null || o.getClass() != this.getClass() ) return false;
- //转换参数的类型
- MyObject that = (MyObject)o;
- //完成所需域的比较
- return ...;
- }
尽管在参考资料[2]中提到,如果有以下2方面的原因使得我们可以使用instanceof代替getClass():
1) 如果需要匹配超类; Java的多态性, 可以将一个子类的实例赋值给一个超类的实例实例
2) "null instanceof [type]"总是返回false,隐含了aThat == null检查(参见 Effective Java by Joshua Bloch.)
也就是说用if ( !(aThat instanceof MyObject) ) return false; 来代替if ( aThat == null || aThat.getClass() != this.getClass() ) return false;
但是在实际应用时应该加以小心,比如MyObject是超类,而MySubObject是子类,看下面的代码:
- public class MyObject
- {
- public int x;
- public MyObject(int x){this.x = x;}
- public boolean equals(Object o)
- {
- if(this == o)return true;
- if(!(o instanceof MyObject))return false;
- return ((MyObject)o).x == x;
- }
- }
- public class MySubObject extends MyObject
- {
- public int x;
- public String text;
- public MySubObject(int x, String text)
- {
- super(x);
- this.x = x;
- this.text = text;
- }
- public boolean equals(Object o)
- {
- if(this == o)return true;
- if(!(o instanceof MySubObject))return false;
- MySubObject obj = (MySubObject)o;
- return (x == obj.x && text.equals(obj.text));
- }
- }
测试代码如下:
- MyObject o1 = new MyObject(1);
- MyObject o2 = new MySubObject(1, "MySubObject's instance");
- System.out.println(o1.equals(o2));
- System.out.println(o2.equals(o1));
尽管我们都知道这2个测试都应该输出false,因为o2实际上是子类的实例的引用,而子类除了比较int数据成员外,还需要比较String数据成员,而超类仅需比较int数据成员;但是结果发现第1个测试输出true,而第二个测试输出false。因为第一个测试调用的是超类的equals方法,并且子类的实例o2匹配了超类,因此返回true。
由此可见,在实现equals方法时最好还是使用getClass()进行比较完成实例到类的匹配更为可靠。
每当覆盖 equals() 时,也应该覆盖 hashCode(),以便将类的实例插到散列表中。hashCode() 应返回一个整数值,并遵循的绝对原则是,它必须返回:
1) 同一对象的相同值。
2) 相等对象的相等值。
每个对象都有一个默认的散列码,就是对象的地址。覆盖hashCode方法的返回值,一般来说,可以返回一个域的hashcode或多个域的hashcode与常量相乘后求和,也可以是将实例域转换为String并组合它们,然后返回结果String的散列码。
个人认为,至于是不是要覆盖hashCode要具体问题具体分析,如果在比较相等时需要比较hashcode则要覆盖,否则无需覆盖;比如集合类有contains方法,contains方法在实现时如果也比较了hashcode,则一定要覆盖hashCode方法;另一个例子是,如果将类的实例作为key放到散列表中,则应该覆盖hashCode方法。当然,同时覆盖hashCode是一个好的习惯。
参考资料:
1. Implementing equals, http://www.javapractices.com/topic/TopicAction.do?Id=17
2. Item 6 :按照常规去覆盖equals方法, http://www.douban.com/subject/discussion/1106608/
3. Java编程介绍--增强对象--覆盖equals(), http://www.java3z.com/cwbwebhome/article/article4/del/j-intjava/j-intjava-10-10.html
4. JAVA指导:深入equals方法, http://www.examda.com/JAVA/Instructs/060522/154004211.html
5. JAVA2核心技术第1卷:基础知识(第7版)
相关推荐
2.javaequals()方法.zip2.javaequals()方法.zip2.javaequals()方法.zip2.javaequals()方法.zip2.javaequals()方法.zip2.javaequals()方法.zip2.javaequals()方法.zip2.javaequals()方法.zip2.javaequals()方法.zip2....
重载equals方法示例重载equals方法示例重载equals方法示例重载equals方法示例重载equals方法示例
Java equals 方法与hashcode 方法的深入解析.rar
Java中equals方法隐藏的陷阱
Java中的==与equals()实例方法Java中测试两个变量是否相等的方法有两个,一个是用==运算符,另一个就是object类提供的equals()方法。2
本文讲述了什么时候重写equals方法和如何重写equals方法。
java_equals用法,用来熟悉重写equals方法的
主要介绍了Java equals()方法使用详解及总结的相关资料,需要的朋友可以参考下
主要介绍了Java中equals()方法重写实现代码的相关资料,需要的朋友可以参考下
java中equals和==的区别.doc java中equals和==的区别.doc
能够加强对java中equals与==区别的理解。
equals()方法和hashCode()方法 270 数据结构 273 Array方法类汇总 304 Java数组与集合小结 305 递归 309 对象的序列化 310 Java两种线程类:Thread和Runnable 315 Java锁小结 321 java.util.concurrent.locks包下...
equals方法重写作业,Students类,有三个属性id ,name ,grade。还有一个测试类用于测试创建了三个对象前两各对象的数据完全一样,第三个对象的数据不同。使用equals方法比较。并输出结果。
Java重写equals同时需要重写hashCode的代码说明,以及如何重写hashCode方法,此代码演示按照effective java书籍说明的重写思路。代码中演示了使用集合存储对象,并且对象作为key,需重写equals和hashCode.
Java语言深入_equals
本文档详细介绍了set接口为什么会用到hashCode和equals方法以及这两个方法的一些探讨 set不同的实现类用到的这两个方法也不同
equals()和hashcode()这两个方法都是从object类中继承过来的。当String 、Math、还有Integer、Double。。。。等这些封装类在使用equals()方法时,已经覆盖了object类的equals()方法.
主要介绍了Java编程中的equals方法使用全解,是Java入门学习中的基础知识,需要的朋友可以参考下
本文介绍了Java语言不直接支持关联数组,可以使用任何对象作为一个索引的数组,但在根Object类中使用 hashCode()方法明确表示期望广泛使用HashMap。理想情况下基于散列的容器提供有效插入和有效检索;直接在对象模式...
java 资料 equals 与== 的区别