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"添加到字符串池中,然後再返回它的引用。 )