1. 程式人生 > >淺談java中的"=="和eqals區別

淺談java中的"=="和eqals區別

short copy 覆寫 main 否則 變量 程序 bar gif

在初學Java時,可能會經常碰到下面的代碼:

1 String str1 = new String("hello");
2 String str2 = new String("hello");
3         
4 System.out.println(str1==str2);
5 System.out.println(str1.equals(str2));

為什麽第4行和第5行的輸出結果不一樣?==和equals方法之間的區別是什麽?如果在初學Java的時候這個問題不弄清楚,就會導致自己在以後編寫代碼時出現一些低級的錯誤。今天就來一起了解一下==和equals方法的區別之處。

java中的數據類型,可分為兩類:
1.基本數據類型,也稱原始數據類型。byte,short,char,int,long,float,double,boolean
他們之間的比較,應用雙等號(==),比較的是他們的值。
2.復合數據類型(類)
當他們用(==)進行比較的時候,比較的是他們在內存中的存放地址,所以,除非是同一個new出來的對象,他們的比較後的結果為true,否則比較後結果為false。 JAVA當中所有的類都是繼承於Object這個基類的,在Object中的基類中定義了一個equals的方法,這個方法的初始行為是比較對象的內存地 址,但在一些類庫當中這個方法被覆蓋掉了,如String,Integer,Date在這些類當中equals有其自身的實現,而不再是比較類在堆內存中的存放地址了。

對於復合數據類型之間進行equals比較,在沒有覆寫equals方法的情況下,他們之間的比較還是基於他們在內存中的存放位置的地址值的,因為Object的equals方法也是用雙等號(==)進行比較的,所以比較後的結果跟雙等號(==)的結果相同。

在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")  

一、認識 "=="

對於復合數據類型之間進行equals比較,在沒有覆寫equals方法的情況下,他們之間的比較還是基於他們在內存中的存放位置的地址值的,因為Object的equals方法也是用雙等號(==)進行比較的,所以比較後的結果跟雙等號(==)的結果相同。

技術分享圖片
 1 public class TestString {
2 public static void main(String[] args) {
3 String s1 = "Monday";
4 String s2 = "Monday";
5 if (s1 == s2)
6 {
7 System.out.println("s1 == s2");}
8 else{
9 System.out.println("s1 != s2");}
10 }
11 }
技術分享圖片 編譯並運行程序,輸出:s1 == s2說明:s1 與 s2 引用同一個 String 對象 -- "Monday"!
2.再稍微改動一下程序,會有更奇怪的發現:
技術分享圖片
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = new String("Monday");
if (s1 == s2)
{System.out.println("s1 == s2");}
else
{System.out.println("s1 != s2");}
if (s1.equals(s2)) {System.out.println("s1 equals s2");}
else{
System.out.println("s1 not equals s2");}
}
}
技術分享圖片
我們將s2用new操作符創建
程序輸出:
s1 != s2
s1 equals s2
說明:s1 s2分別引用了兩個"Monday"String對象

3. 字符串緩沖池
原來,程序在運行的時候會創建一個字符串緩沖池當使用 s2 = "Monday" 這樣的表達是創建字符串的時候,程序首先會在這個String緩沖池中尋找相同值的對象,在第一個程序中,s1先被放到了池中,所以在s2被創建的時候,程序找到了具有相同值的 s1
將s2引用s1所引用的對象"Monday"
第二段程序中,使用了 new 操作符,他明白的告訴程序:"我要一個新的!不要舊的!"於是一個新的"Monday"Sting對象被創建在內存中。

對於非基本數據類型的變量,在一些書籍中稱作為 引用類型的變量。比如上面的str1就是引用類型的變量,引用類型的變量存儲的並不是 “值”本身,而是於其關聯的對象在內存中的地址。比如下面這行代碼:

  String str1;

  這句話聲明了一個引用類型的變量,此時它並沒有和任何對象關聯。

  而 通過new String("hello")來產生一個對象(也稱作為類String的一個實例),並將這個對象和str1進行綁定:

  str1= new String("hello");

  那麽str1指向了一個對象(很多地方也把str1稱作為對象的引用),此時變量str1中存儲的是它指向的對象在內存中的存儲地址,並不是“值”本身,也就是說並不是直接存儲的字符串"hello"。這裏面的引用和C/C++中的指針很類似。

  因此在用==對s2和s1進行比較時,得到的結果是false。因此它們分別指向的是不同的對象,也就是說它們實際存儲的內存地址不同。

他們的值相同,但是位置不同,一個在池中遊泳一個在岸邊休息。哎呀,真是資源浪費,明明是一樣的非要分開做什麽呢?

4.再次更改程序:
技術分享圖片
public class TestString {
public static void main(String[] args) {
String s1 = "Monday";
String s2 = new String("Monday");
s2 = s2.intern();
if (s1 == s2)
{System.out.println("s1 == s2");}
else
{System.out.println("s1 != s2");}
if (s1.equals(s2)) {System.out.println("s1 equals s2");}
else{
System.out.println("s1 not equals s2");}
}
}
技術分享圖片
這次加入:s2 = s2.intern();
程序輸出:
s1 == s2
s1 equals s2
原 來,(java.lang.String的intern()方法"abc".intern()方法的返回值還是字符串"abc",表面上看起來好像這個方 法沒什麽用處。但實際上,它做了個小動作:檢查字符串池裏是否存在"abc"這麽一個字符串,如果存在,就返回池裏的字符串;如果不存在,該方法會 把"abc"添加到字符串池中,然後再返回它的引用。

二.equals比較的又是什麽?

  equals方法是基類Object中的方法,因此對於所有的繼承於Object的類都會有該方法。為了更直觀地理解equals方法的作用,直接看Object類中equals方法的實現。

但是有些朋友又會有疑問了,為什麽下面一段代碼的輸出結果是true?

要知道究竟,可以看一下String類的equals方法的具體實現,同樣在該路徑下,String.java為String類的實現。

  下面是String類中equals方法的具體實現:

技術分享圖片

可以看出,String類對equals方法進行了重寫,用來比較指向的字符串對象所存儲的字符串是否相等。

  其他的一些類諸如Double,Date,Integer等,都對equals方法進行了重寫用來比較指向的對象所存儲的內容是否相等。

  總結來說:

  1)對於==,如果作用於基本數據類型的變量,則直接比較其存儲的 “值”是否相等;

    如果作用於引用類型的變量,則比較的是所指向的對象的地址

  2)對於equals方法,註意:equals方法不能作用於基本數據類型的變量

    如果沒有對equals方法進行重寫,則比較的是引用類型的變量所指向的對象的地址;

    諸如String、Date等類對equals方法進行了重寫的話,比較的是所指向的對象的內容。

淺談java中的"=="和eqals區別