java的自動裝箱與拆箱詳解
什麼是自動裝箱拆箱
基本資料型別的自動裝箱(autoboxing)、拆箱(unboxing)是自J2SE 5.0開始提供的功能。
一般我們要建立一個類的物件例項的時候,我們會這樣:
Class a = new Class(parameter);
當我們建立一個Integer物件時,卻可以這樣:
Integer i = 100; (注意:不是 int i = 100; )
實際上,執行上面那句程式碼的時候,系統為我們執行了:Integer i = Integer.valueOf(100);
此即基本資料型別的自動裝箱功能。
基本資料型別與物件的差別
基本資料型別不是物件,也就是使用int、double、boolean等定義的變數、常量。
基本資料型別沒有可呼叫的方法。
eg: int t = 1; t. 後面是沒有方法滴。
Integer t =1; t. 後面就有很多方法可讓你呼叫了。
什麼時候自動裝箱
例如:Integer i = 100;
相當於編譯器自動為您作以下的語法編譯:Integer i = Integer.valueOf(100);
什麼時候自動拆箱
自動拆箱(unboxing),也就是將物件中的基本資料從物件中自動取出。如下可實現自動拆箱:
Integer i = 10; //裝箱
int t = i; //拆箱,實際上執行了 int t = i.intValue();
在進行運算時,也可以進行拆箱。
Integer i = 10; System.out.println(i++);
Integer的自動裝箱
//在-128~127 之外的數
Integer i1 =200;
Integer i2 =200;
System.out.println("i1==i2: "+(i1==i2));
// 在-128~127 之內的數
Integer i3 =100;
Integer i4 =100;
System.out.println("i3==i4: "+(i3==i4));
輸出的結果是:
i1==i2: false
i3==i4: true
說明:
equals() 比較的是兩個物件的值(內容)是否相同。
“==” 比較的是兩個物件的引用(記憶體地址)是否相同,也用來比較兩個基本資料型別的變數值是否相等。
前面說過,int 的自動裝箱,是系統執行了 Integer.valueOf(int i),先看看Integer.java的原始碼:
public static Integer valueOf(int i) {
if(i >= -128 && i <= IntegerCache.high) // 沒有設定的話,IngegerCache.high 預設是127
return IntegerCache.cache[i + 128];
else
return new Integer(i);
}
對於–128到127(預設是127)之間的值,Integer.valueOf(int i) 返回的是快取的Integer物件(並不是新建物件)
所以範例中,i3 與 i4實際上是指向同一個物件。
而其他值,執行Integer.valueOf(int i) 返回的是一個新建的 Integer物件,所以範例中,i1與i2 指向的是不同的物件。
當然,當不使用自動裝箱功能的時候,情況與普通類物件一樣,請看下例:
Integer i3 =new Integer(100);
Integer i4 =new Integer(100);
System.out.println("i3==i4: "+(i3==i4));//顯示false
String 的拆箱裝箱
先看個例子:
String str1 ="abc";
String str2 ="abc";
System.out.println(str2==str1); //輸出為 true
System.out.println(str2.equals(str1)); //輸出為 true
String str3 =new String("abc");
String str4 =new String("abc");
System.out.println(str3==str4); //輸出為 false
System.out.println(str3.equals(str4)); //輸出為 true
這個怎麼解釋呢?貌似看不出什麼。那再看個例子。
String d ="2";
String e ="23";
e = e.substring(0, 1);
System.out.println(e.equals(d)); //輸出為 true
System.out.println(e==d); //輸出為 false
第二個例子中,e的初始值與d並不同,因此e與d是各自建立了個物件,(e==d)為false 。
同理可知,第一個例子中的str3與str4也是各自new了個物件,而str1與str2卻是引用了同一個物件。