Java語法----Java中equals和==的區別

平時在學Android和Java語言的時候,老是碰到「equals」和「==」這兩個字符,老感受差很少;其實仍是有一些區別的,今天干脆把它們完全弄清楚。 java

1、java當中的數據類型和「==」的含義: 面試

  • 基本數據類型(也稱原始數據類型) :byte,short,char,int,long,float,double,boolean。他們之間的比較,應用雙等號(==),比較的是他們的值
  • 引用數據類型:當他們用(==)進行比較的時候,比較的是他們在內存中的存放地址(確切的說,是堆內存地址)。

注:對於第二種類型,除非是同一個new出來的對象,他們的比較後的結果爲true,不然比較後結果爲false。由於每new一次,都會從新開闢堆內存空間ide

 

2、equals()方法介紹: this

JAVA當中全部的類都是繼承於Object這個超類的,在Object類中定義了一個equals的方法,equals的源碼是這樣寫的: spa

public boolean equals(Object obj) { //this - s1 //obj - s2 return (this == obj); }

能夠看到,這個方法的初始默認行爲是比較對象的內存地址值,通常來講,意義不大。因此,在一些類庫當中這個方法被重寫了,如String、Integer、Date。在這些類當中equals有其自身的實現(通常都是用來比較對象的成員變量值是否相同),而再也不是比較類在堆內存中的存放地址了。 
因此說,對於複合數據類型之間進行equals比較,在沒有覆寫equals方法的狀況下,他們之間的比較仍是內存中的存放位置的地址值,跟雙等號(==)的結果相同;若是被複寫,按照複寫的要求來code

咱們對上面的兩段內容作個總結吧: 對象

 == 的做用:
  基本類型:比較的就是值是否相同
  引用類型:比較的就是地址值是否相同
equals 的做用:
  引用類型:默認狀況下,比較的是地址值。
注:不過,咱們能夠根據狀況本身重寫該方法。通常重寫都是自動生成,比較對象的成員變量值是否相同 繼承

 

 

3、String類的equals()方法: ip

如今咱們拿String類來舉例: 內存

咱們去\src\java\lang目錄中找到String類,發現equals方法被重寫以下:

複製代碼
 1 public boolean equals(Object anObject) {  2 if (this == anObject) {  3 return true;  4  }  5 if (anObject instanceof String) {  6 String anotherString = (String)anObject;  7 int n = value.length;  8 if (n == anotherString.value.length) {  9 char v1[] = value; 10 char v2[] = anotherString.value; 11 int i = 0; 12 while (n-- != 0) { 13 if (v1[i] != v2[i]) 14 return false; 15 i++; 16  } 17 return true; 18  } 19  } 20 return false; 21 }
複製代碼

上述代碼能夠看出,String類中被複寫的equals()方法實際上是比較兩個字符串的內容。下面咱們經過實際代碼來看看String類的比較。

一、舉例代碼以下:

複製代碼
1 public class StringDemo { 2 public static void main(String[] args) { 3 String s1 = "Hello"; 4 String s2 = "Hello"; 5 System.out.println(s1 == s2); // true 6  } 7 }
複製代碼

上方代碼中,用「==」比較s1和s2,返回的結果是true。

二、稍微改動一下程序,會有奇怪的發現:

複製代碼
 1 public class StringDemo {  2 public static void main(String args[]) {  3 String str1 = "Hello";  4 String str2 = new String("Hello");  5 String str3 = str2; // 引用傳遞  6 System.out.println(str1 == str2); // false  7 System.out.println(str1 == str3); // false  8 System.out.println(str2 == str3); // true  9 System.out.println(str1.equals(str2)); // true 10 System.out.println(str1.equals(str3)); // true 11 System.out.println(str2.equals(str3)); // true 12  } 13 }
複製代碼

 上方第4行代碼中,咱們new了一個對象,用「==」比較s1和s2,返回的結果倒是false;而用用「equals」比較s1和s2,返回的結果是true。 

爲了分析上面的代碼,咱們必須首先分析堆內存空間和棧內存空間,這一點很是重要:

看完上面的圖,再結合上面的代碼,就一目瞭然了。如今咱們能夠給本身出一道面試題:

面試題:請解釋字符串比較之中「==」和equals()的區別?

  •  ==:比較的是兩個字符串內存地址(堆內存)的數值是否相等,屬於數值比較;
  •  equals():比較的是兩個字符串的內容,屬於內容比較。

之後進行字符串相等判斷的時候都使用equals()。

三、再次更改程序:

複製代碼
1 public class ObjectDemo{ 2 public static void main(String[] args) { 3 String s1 = "Hello"; 4 String s2 = new String("Hello"); 5 s2 = s2.intern(); 6 System.out.println(s1 == s2); // true 7 System.out.println(s1.equals(s2)); // true 8  } 9 }
複製代碼

上述代碼的第5行中,java.lang.String的intern()方法"abc".intern()方法的返回值仍是字符串"abc",表面上看起來好像這個方法沒什麼用處。但實際上,它作了個小動做:檢查字符串池裏是否存在"abc"這麼一個字符串,若是存在,就返回池裏的字符串;若是不存在,該方法會 把"abc"添加到字符串池中,而後再返回它的引用。

 

4、比較兩個對象的值:

代碼以下:

複製代碼
 1 package com.smyh;  2  3 public class ObjectDemo {  4 public static void main(String args[]){  5 Student student1 = new Student("生命壹號",22,"成都");  6 Student student2 = new Student("生命壹號",22,"成都");  7 System.out.println(student1==student2);  8  System.out.println(student1.equals(student2));  9  } 10 } 11 class Student { 12 private String name; 13 private int age; 14 private String address; 15 public Student(String name,int age,String address){ 16 this.name = name; 17 this.age = age; 18 this.address = address; 19  } 20 //重寫Object類中的equals方法(比較兩個對象的值是否相等) 21 public boolean equals(Object obj){ 22 //爲了提升效率:若是兩個內存地址相等,那麼必定是指向同一個對內存中的對象,就無需比較兩個對象的屬性值(本身跟本身比,沒啥意義嘛) 23 if(this==obj){ 24 return true; 25  } 26 27 //爲了提供程序的健壯性 28 //我先判斷一下,obj是否是學生的一個對象,若是是,再作向下轉型,若是不是,直接返回false。 29 //這個時候,咱們要判斷的是對象是不是某個類的對象? 30 //記住一個格式:對象名 instanceof 類名。表示:判斷該對象是不是該類的一個對象  31 if(!(obj instanceof Student)){ 32 return false; 33  } 34 35 //若是是就繼續 36 Student s = (Student)obj;//強制轉換,即向下轉型(畢竟Object類型沒有具體的對象屬性)  37 return this.name.equals(s.name) && this.age == s.age && this.address.equals(s.address);//判斷兩個對象的屬性值是否相等 38  } 39 }
複製代碼

上述代碼中,首先判斷傳遞進來的對象與當前對象的地址是否相等,若是相等,則確定是同一個堆內存中的對象。由於傳遞進來的參數是Object類型,因此任何對象均可以接收。一旦接收進來,就將Object類型的對象向下轉型,而後判斷具體的屬性值。

 運行效果:

其實,若是是在Eclipse中作開發的話,上面重寫的equals()方法實際上是能夠自動生成的:

自動生成後的equals()方法以下:

複製代碼
 1  @Override  2 public boolean equals(Object obj) {  3 if (this == obj)  4 return true;  5 if (obj == null)  6 return false;  7 if (getClass() != obj.getClass())  8 return false;  9 Student other = (Student) obj; 10 if (address == null) { 11 if (other.address != null) 12 return false; 13 } else if (!address.equals(other.address)) 14 return false; 15 if (age != other.age) 16 return false; 17 if (name == null) { 18 if (other.name != null) 19 return false; 20 } else if (!name.equals(other.name)) 21 return false; 22 return true; 23 }
複製代碼

 

能夠看到:咱們本身重寫的equals()方法和自動生成的equals()方法,兩者效果是同樣的。

相關文章
相關標籤/搜索