分析:“==”,equals和hashCode有什么区别

“==”,equals和hashCode有什么区别,前两者都是在开发场景中用到的东西,后者在面试的时候问的比较多,前后两者都是香铺相成的。

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

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

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

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

我们来看一段代码:

public class DemoTest {
      public static void main(String[] args) {
          String s = new String("Hello");
          String s2 = new String("Hello2");
          System.out.println(s == s2);
          System.out.println(s.endsWith(s2));
      }
  }
 //输出打印
E:\java\bin\java.exe "-javaagent:D:\Program Files\JetBrains\IntelliJ IDEA lang\2.6\commons-lang-2.6.jar com.mayikt.handler.controller.DemoTest
false
true

两个new语句创建了两个对象,然后用s1,s2这两个变量分别指向一个对象,这是两个不同的对象,它们的首地址是不同的。

既a和b中存储的数值是不同的,所以a和b将会返回fasle,而这两个对象中的内容是相同的,所以表达式a.equals(b)将返回true,

如果一个类没有自己定义equals()方法,那么他将继承Object类的equals()方法,

boolean equals(Object o){
      return this == 0;
}

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

下面我们来说说hashCode()方法,

首先,我们都知道hashCode()方法是从Object类中继承过来的,它也用来鉴定两个对象是否相等,Object类中的hashCode()方法

返回对象在内存中地址转换成的一个int值,所以如果没有重写hashCode()方法,任何对象的hashCode()方法都是不相等的。

虽然equals()方法也是用来判断两个对象是否相等的,但是它与hashCode()方法是有区别的,一般来讲,equals()方法是给用户调用的,如果需要判断两个对象是否相等,可以重写equals()方法,然后在代码中调用,这样就可以判断他们是否相等了,对于hashCode()方法,用户一般不会去调用它,列如在hashMap中,由于key是不可以重复的,他在判断key是否重复时,就判断了hashCode()这个方法,而且也用到了equals()方法,此外,不可以重复指的是equals()和hashCode()方法只要有一个不等就可以了,所以,hahCode()方法相当于是一个对象的编码,就好象文件中的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()方法的返回值则可能相等,也可能不相等。

发布了41 篇原创文章 · 获赞 8 · 访问量 1293

猜你喜欢

转载自blog.csdn.net/sdgames/article/details/104986793