所有的对象都有标识(内存中的地址)和状态(对象的数据)。'=='运算符比较两个对象的地址,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是一个好的习惯。 |
|