A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© 果维 中级黑马   /  2015-12-27 20:05  /  233 人查看  /  0 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

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

0 个回复

您需要登录后才可以回帖 登录 | 加入黑马