1. 程式人生 > >java的自動裝箱與拆箱詳解

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卻是引用了同一個物件。