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方法也是用雙等號( ==)進行比較的,所以比較後的結果跟雙等號(==)的結果相同。
複製代碼代碼如下:
publicclass TestString {
publicstaticvoid main(String[] args) {
String s1 ="Monday";
String s2 ="Monday";
if (s1 == s2)
{
System.out.println("s1 == s2");}
else{
System.out.println("s1 != s2");}
}
}
編譯並執行程序,輸出:s1 == s2說明:s1 與s2 引用同一個String 物件-- "Monday"! 2.再稍微改動一下程序,會有更奇怪的發現:
複製代碼代碼如下:
publicclass TestString {
publicstaticvoid 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對像被創建在內存中。他們的值相同,但是位置不同,一個在池中游泳一個在岸邊休息。哎呀,真是資源浪費,明明是一樣的非要分開做什麼呢?
4.再次更改程序:
複製代碼代碼如下:
publicclass TestString
{
publicstaticvoid 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"加到字串池中,然後再回傳它的引用。