1.先上案例:java
public class TestExecutor { public static void main(String[] args) { // TODO Auto-generated method stub String str1 = new String("hello"); String str2 = new String("hello"); System.out.println(str1 == str2); System.out.println(str1.equals(str1)); } }
運行後打印狀況:編程
爲何這二者打印的狀況會不一樣,今天咱們就帶着這個問題來看一下equals()與「==」到底有什麼區別?spa
下面這個句話是摘自《Java編程思想》一書中的原話:指針
「關係操做符生成的是一個boolean結果,它們計算的是操做數的值之間的關係」。code
這句話看似簡單,理解起來仍是須要細細體會的。說的簡單點,==就是用來比較值是否相等。下面先看幾個例子:對象
public class TestExecutor { public static void main(String[] args) { // TODO Auto-generated method stub int n=4; int m=4; System.out.println(n==m); String str = new String("hello"); String str1 = new String("hello"); String str2 = new String("hello"); System.out.println(str1==str2); str1 = str; str2 = str; System.out.println(str1==str2); } }
打印狀況:true, false, true繼承
n==m結果爲true,這個很容易理解,變量n和變量m存儲的值都爲3,確定是相等的。而爲何str1和str2兩次比較的結果不一樣?要理解這個其實只須要理解基本數據類型變量和非基本數據類型變量的區別。ip
在Java中游8種基本數據類型:內存
浮點型:float(4 byte), double(8 byte)字符串
整型:byte(1 byte), short(2 byte), int(4 byte) , long(8 byte)
字符型: char(2 byte)
布爾型: boolean(JVM規範沒有明確規定其所佔的空間大小,僅規定其只可以取字面值"true"和"false")
對於這8種基本數據類型的變量,變量直接存儲的是「值」,所以在用關係操做符==來進行比較時,比較的就是 「值」 自己。要注意浮點型和整型都是有符號類型的,而char是無符號類型的(char類型取值範圍爲0~2^16-1).
也就是說好比:
int n=4; int m=4;
變量n和變量m都是直接存儲的"3"這個數值,因此用==比較的時候結果是true。
而對於非基本數據類型的變量,在一些書籍中稱做爲 引用類型的變量。好比上面的str1就是引用類型的變量,引用類型的變量存儲的並非 「值」自己,而是於其關聯的對象在內存中的地址。好比下面這行代碼:
String str1;
這句話聲明瞭一個引用類型的變量,此時它並無和任何對象關聯。
而 經過new String("hello")來產生一個對象(也稱做爲類String的一個實例),並將這個對象和str1進行綁定:
str1= new String("hello");
那麼str1指向了一個對象(不少地方也把str1稱做爲對象的引用),此時變量str1中存儲的是它指向的對象在內存中的存儲地址,並非「值」自己,也就是說並非直接存儲的字符串"hello"。這裏面的引用和C/C++中的指針很相似。
所以在用==對str1和str2進行第一次比較時,獲得的結果是false。所以它們分別指向的是不一樣的對象,也就是說它們實際存儲的內存地址不一樣。
而在第二次比較時,都讓str1和str2指向了str指向的對象,那麼獲得的結果毫無疑問是true。
equals方法是基類Object中的方法,所以對於全部的繼承於Object的類都會有該方法。爲了更直觀地理解equals方法的做用,直接看Object類中equals方法的實現。
該類的源碼路徑爲:D:\Program Files\Java\jdk1.6.0_14的src.zip 的java.lang路徑下的Object.java(視我的jdk安裝路徑而定)。
下面是Object類中equals方法的實現:
很顯然,在Object類中,equals方法是用來比較兩個對象的引用是否相等,便是否指向同一個對象。
可是有些朋友又會有疑問了,爲何下面一段代碼的輸出結果是true?
public class TestMain { /** * @param args */ public static void main(String[] args) { // TODO Auto-generated method stub String str1 = new String("hello"); String str2 = new String("hello"); System.out.println(str1.equals(str2)); } }
要知道究竟,能夠看一下String類的equals方法的具體實現,一樣在該路徑下,String.java爲String類的實現。
下面是String類中equals方法的具體實現:
能夠看出,String類對equals方法進行了重寫,用來比較指向的字符串對象所存儲的字符串是否相等。
其餘的一些類諸如Double,Date,Integer等,都對equals方法進行了重寫用來比較指向的對象所存儲的內容是否相等。
總結來講:
1)對於==,若是做用於基本數據類型的變量,則直接比較其存儲的 「值」是否相等;
若是做用於引用類型的變量,則比較的是所指向的對象的地址
2)對於equals方法,注意:equals方法不能做用於基本數據類型的變量
若是沒有對equals方法進行重寫,則比較的是引用類型的變量所指向的對象的地址;
諸如String、Date等類對equals方法進行了重寫的話,比較的是所指向的對象的內容。