Java程序员面试笔试宝典(第2版)
上QQ阅读APP看书,第一时间看更新

1.17 “==”、equals和hashCode的区别

1)“==”运算符用来比较两个变量的值是否相等,也就是用于比较变量所对应的内存中存储的数值是否相同,要比较两个基本类型的数据或两个引用变量是否相等,只能用“==”运算符。

具体而言,如果两个变量是基本数据类型,可以直接用“==”来比较其对应的值是否相等。如果一个变量指向的数据是对象(引用类型),那么,此时涉及了两块内存,对象本身占用一块内存(堆内存),对象的引用也占用一块内存。例如,对于赋值语句String s=new String(),变量s占用一块存储空间(一般在栈中),而new String()则存储在另外一块存储空间里(一般在堆中),此时,变量s所对应的内存中存储的数值就是对象占用的那块内存的首地址。对于指向对象类型的变量,如果要比较两个变量是否指向同一个对象,即要看这两个变量所对应的内存中的数值是否相等(这两个对象是否指向同一块存储空间),这时候就可以用“==”运算符进行比较。但是,如果要比较这两个对象的内容是否相等,那么用“==”运算符就无法实现了。

2)equals是Object类提供的方法之一,每一个Java类都继承自Object类,所以每一个对象都具有equals这个方法。Object类中定义的equals(Object)方法是直接使用“==”比较的两个对象,所以在没有覆盖equals(Object)方法的情况下,equals(Object)与“==”一样,比较的是引用。

相比“==”运算符,equals(Object)方法的特殊之处就在于它可以被覆盖,所以可以通过覆盖这个方法让它不是比较引用而是比较对象的属性。例如String类的equals方法是用于比较两个独立对象的内容是否相同,即堆中的内容是否相同。例如,对于下面的代码:

两条new语句在堆中创建了两个对象,然后用s1、s2这两个变量分别指向这两个对象,这是两个不同的对象,它们的首地址是不同的,即s1和s2中存储的数值是不相同的,所以,表达式a==b将返回false,而这两个对象中的内容是相同的,所以,表达式a.equals(b)将返回true。

如果一个类没有实现equals方法,那么它将继承Object类的equals方法,Object类的equals方法的实现代码如下:

通过以上例子可以说明,如果一个类没有自己定义equals方法,它默认的equals方法(从Object类继承的)就是使用“==”运算符,也就是在比较两个变量指向的对象是否是同一对象,此时使用equals方法和使用“==”会得到同样的结果,如果比较的是两个独立的对象则总返回false。如果编写的类希望能够比较该类创建的两个实例对象的内容是否相同,那么必须覆盖equals方法,由开发人员自己写代码来决定在什么情况下即可认为两个对象的内容是相同的。

3)hashCode()方法是从Object类中继承过来的,它也用来鉴定两个对象是否相等。Object类中的hashCode()方法返回对象在内存中地址转换成的一个int值,所以如果没有重写hashCode()方法,任何对象的hashCode()方法都是不相等的。

虽然equals方法也是用来判断两个对象是否相等的,但是二者是有区别的。一般来讲,equals方法是给用户调用的,如果需要判断两个对象是否相等,可以重写equals方法,然后在代码中调用,就可以判断它们是否相等了。对于hashCode()方法,用户一般不会去调用它,例如在HashMap中,由于key是不可以重复的,它在判断key是否重复的时候就判断了hashCode()这个方法,而且也用到了equals方法。此处“不可以重复”指的是equals和hashCode()只要有一个不等就可以了。所以,hashCode()相当于是一个对象的编码,就好像文件中的md5,它与equals方法的不同之处就在于它返回的是int型,比较起来不直观。

一般在覆盖equals方法的同时也要覆盖hashCode()方法,否则,就会违反Object.hashCode的通用约定,从而导致该类无法与所有基于散列值(hash)的集合类(HashMap、HashSet和Hashtable)结合在一起正常运行。

hashCode()的返回值和equals方法的关系如下:如果x.equals(y)返回true,即两个对象根据equals方法比较是相等的,那么调用这两个对象中任意一个对象的hashCode()方法都必须产生同样的整数结果。如果x.equals(y)返回false,即两个对象根据equals()方法比较是不相等的,那么x和y的hashCode()方法的返回值有可能相等,也有可能不等。反过来,hashCode()方法的返回值不等,一定能推出equals方法的返回值也不等,而hashCode()方法的返回值相等,equals方法的返回值则可能相等,也可能不等。