《Thinking in Java》读书笔记(三)——equals方法

Equals()方法用于比较对象的内容还是引用?

(中文书P45英文书P104中用到)                                                                                   

 

EqualsMethod.java中意思是说equals()比较的是两个对象的实际内容而不是引用,而在Equalsmethod2.java里,却显示equals()比较的也是引用,后面更明说“equals()的默认行为是比较引用”,看得我一头雾水,到底equals()比较的是对象的内容还是对象的引用呢?

 

接着,找到一篇《深入研究Java equals方法》的文章,觉得写得很好,挺详细的,转载过来以便日后查看。

 

equals方法的重要性毋须多言,只要你想比较两个对象是不是同一对象,你就应该实现equals方法,让对象用你认为相等的条件来进行比较.

     下面的内容只是API的规范,没有什么太高深的意义,但我之所以最先把它列在这儿,是因为这些规范在事实中并不是真正能保证得到实现.
1.
对于任何引用类型, o.equals(o) == true成立
.

2.
如果 o.equals(o1) == true 成立,那么o1.equals(o)==true也一定要成立
.

3.
如果 o.equals(o1) == true 成立且  o.equals(o2) == true 成立,那么

  o1.equals(o2) == true
也成立.

4.
如果第一次调用o.equals(o1) == true成立,在oo1没有改变的情况下以后的任何次调用都成立
.

5.o.equals(null) == true
任何时间都不成立
.

     以上几条规则并不是最完整的表述,详细的请参见API文档.对于Object,它提供了一个最最严密的实现,那就是只有是同一对象时,equals方法才返回true,也就是人们常说的引用比较而不是值比较.这个实现严密得已经没有什么实际的意义, 所以在具体子类(相对于Object来说),如果我们要进行对象的值比较,就必须实现自己的equals方法.先来看一下以下这段程序:

    public boolean equals(Object obj)
    {
        if (obj == null) return false;
        if (!(obj instanceof FieldPosition))
            return false;
        FieldPosition other = (FieldPosition) obj;
        if (attribute == null) {
            if (other.attribute != null) {
                return false;
            }
        }
        else if (!attribute.equals(other.attribute)) {
            return false;
        }
        return (beginIndex == other.beginIndex
            && endIndex == other.endIndex
            && field == other.field);
    } 

     
这是JDKjava.text.FieldPosition的标准实现,似乎没有什么可说的我信相大多数或绝大多数程序员认为,这是正确的合法的equals实现.毕竟它是JDKAPI实现啊还是让我们以事实来说话吧
:

package debug
;import java.text.*;
public class Test {
  public static void main(String[] args) {
    FieldPosition fp = new FieldPosition(10);
    FieldPosition fp1 = new MyTest(10);
    System.out.println(fp.equals(fp1));
    System.out.println(fp1.equals(fp));
  }
}
class MyTest extends FieldPosition{
  int x = 10;
  public MyTest(int x){
    super(x);
    this.x = x;
  }
  public boolean equals(Object o){
    if(o==null) return false;
    if(!(o instanceof MyTest )) return false;
    return ((MyTest)o).x == this.x;
  }

}

运行一下看看会打印出什么
:
System.out.println(fp.equals(fp1));
打印
true
System.out.println(fp1.equals(fp));
打印
flase

     
两个对象,出现了不对称的equals算法.问题出在哪里(脑筋急转弯:当然出在JDK实现的BUG)?我相信有太多的程序员(除了那些根本不知道实现equals方法的程序员外)在实现equals方法时都用过instanceof运行符来进行短路优化的,实事求是地说很长一段时间我也这么用过。


     
太多的教程,文档都给了我们这样的误导。而有些稍有了解的程序员可能知道这样的优化可能有些不对但找不出问题的关键。另外一种极端是知道这个技术缺陷的骨灰级专家就提议不要这样应用。我们知道,"通常"要对两个对象进行比较,那么它们"应该"是同一类型。所以首先利用instanceof运算符进行短路优化,如果被比较的对象不和当前对象是同一类型则不用比较返回false

    但事实上,"子类是父类的一个实例",所以如果子类 o instanceof 父类,始终返回true,这时肯定不会发生短路优化,下面的比较有可能出现多种情况,一种是不能造型成父类而抛出异常,另一种是父类的private 成员没有被子类继承而不能进行比较,还有就是形成上面这种不对称比较。可能会出现太多的情况。

    
那么,是不是就不能用 instanceof运算符来进行优化?答案是否定的,JDK中仍然有很多实现是正确的,如果一个classfinal的,明知它不可能有子类,为什么不用 instanceof来优化呢?为了维护SUN的开发小组的声誉,我不说明哪个类中,但有一个小组成员在用这个方法优化时在后加加上了加上了这样的注释:
        if (this == obj)                      // quick check
            return true;
        if (!(obj instanceof XXXXClass))         // (1) same object?
            return false;

     
可能是有些疑问,但不知道如何做(不知道为什么没有打电话给我......)那么对于非final类,如何进行类型的quick check呢?

if(obj.getClass() != XXXClass.class) return false;

     
用被比较对象的class对象和当前对象的class比较,看起来是没有问题,但是,如果这个类的子类没有重新实现equals方法,那么子类在比较的时候,obj.getClass() 肯定不等于XXXCalss.class, 也就是子类的equals将无效,所以

if(obj.getClass() != this.getClass()) return false;

才是正确的比较。另外一个quick checkif(this==obj) return true;

是否equals方法比较的两个对象一定是要同一类型?上面我用了"通常",这也是绝大多数程序员的愿望,但是有些特殊的情况,我们可以进行不同类型的比较,这并不违反规范。但这种特殊情况是非常罕见的,一个不恰当的例子是,Integer类的equals可以和Sort做比较,比较它们的value是不是同一数学值。(事实上JDKAPI中并没有这样做,所以我才说是不恰当的例子)在完成quick check以后,我们就要真正实现你认为的相等。对于如果实现对象相等,没有太高的要求,比如你自己实现的类,你可以认为只要name相同即认为它们是相等的,其它的sex, ago都可以不考虑。这是不完全实现,但是如果是完全实现,即要求所有的属性都是相同的,那么如何实现equals方法?

class Human{
 private String name;
 private int ago;
 private String sex;
        ....................
  public boolean equals(Object obj){
  quick check.......
  Human other = (Human)ojb;
  return this.name.equals(other.name)  && this.ago == ohter.ago  && this.sex.equals(other.sex);
 }
}

这是一个完全实现,但是,有时equals实现是在父类中实现,而要求被子类继承后equals能正确的工
作,这时你并不事实知道子类到底扩展了哪些属性,所以用上面的方法无法使equals得到完全实现。
一个好的方法是利用反射来对equals进行完全实现:  

public boolean equals(Object obj){
  quick check.......
  Class c = this.getClass();
  Filed[] fds = c.getDeclaredFields();
  for(Filed f:fds){
   if(!f.get(this).equals(f.get(obj)))
    return false;
  }
  return true;
 }

为了说明的方便,上明的实现省略了异常,这样的实现放在父类中,可以保证你的子类的equals可以按

你的愿望正确地工作。关于equals方法的最后一点是:如果你要是自己重写(正确说应该是履盖)了equals方法,那同时就一
定要重写hashCode().这是规范,否则.............


我们还是看一下这个例子:

public final class PhoneNumber {
    private final int areaCode;
    private final int exchange;
    private final int extension;  

  public PhoneNumber(int areaCode, int exchange, int extension) {
        rangeCheck(areaCode, 999, "area code");
        rangeCheck(exchange, 99999999, "exchange");
        rangeCheck(extension, 9999, "extension");
        this.areaCode = areaCode;
        this.exchange = exchange;
        this.extension = extension;
    }  

  private static void rangeCheck(int arg, int max, String name) {
        if(arg < 0 || arg > max)
            throw new IllegalArgumentException(name + ": " + arg);
    }  

    public boolean equals(Object o) {
        if(o == this)
            return true;
        if(!(o instanceof PhoneNumber))
            return false;
        PhoneNumber pn = (PhoneNumber)o;
        return pn.extension == extension && pn.exchange == exchange && pn.areaCode == areaCode;
    }
}

注意这个类是final的,所以这个equals实现没有什么问题。我们来测试一下:   

public static void main(String[] args) {
        Map hm = new HashMap();
        PhoneNumber pn = new PhoneNumber(123, 38942, 230);
        hm.put(pn, "I love you");
        PhoneNumber pn1 = new PhoneNumber(123, 38942, 230);
        System.out.println(pn);
        System.out.println("pn.equals(pn1) is " + pn.equals(pn1));
        System.out.println(hm.get(pn1));
        System.out.println(hm.get(pn));
    }

既然pn.equals(pn1),那么我put(pn,"I love you")后,get(pn1)为什么是null呢?

答案是因为它们的hashCode不一样,而hashMap就是以hashCode为主键的。所以规范要求,如果两个对象进行equals比较时如果返回true,那么它们的hashcode要求返回相等的值。

 

资料来源:http://java.ccidnet.com/art/3737/20071113/1274133_1.html

 

 

虽然后面一段看得不是很懂,但是看完也基本找到了我所提出的问题的答案。在Object类的基本实现中,equals方法确实是比较对象的引用的,而书中EqualsMethod.java中的Integer类是覆盖了equals方法的,使其用于比较对象的内容。

而在另一篇文章中,我看到如下的结论:“equals方法对于字符串来说是比较内容的,而对于非字符串来说是比较其指向的对象是否相同的;== 比较符也是比较指向的对象是否相同的也就是对象在对内存中的的首地址。”

可见String类中也覆盖了equals方法。

 

再看一下书,确实是自己太粗心了,没有留意没有理解重要的内容。书中有这么一句话“大多数Java类库都实现了equals()方法,以便用来比较对象的内容,而非比较对象的引用。”这里是说大多数类,但不是所有类,所以在具体使用时还得查JDK文档。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值