国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 學院 > 開發設計 > 正文

深入剖析Java中的裝箱和拆箱

2019-11-14 14:57:55
字體:
來源:轉載
供稿:網友

自動裝箱和拆箱問題是java中一個老生常談的問題了,今天我們就來一些看一下裝箱和拆箱中的若干問題。本文先講述裝箱和拆箱最基本的東西,再來看一下面試筆試中經常遇到的與裝箱、拆箱相關的問題。

  以下是本文的目錄大綱:

  一.什么是裝箱?什么是拆箱?

  二.裝箱和拆箱是如何實現的

  三.面試中相關的問題

  若有不正之處,請諒解和批評指正,不勝感激。

  請尊重作者勞動成果,轉載請標明原文鏈接:

   http://www.survivalescaperooms.com/dolphin0520/p/3780005.html

一.什么是裝箱?什么是拆箱?

  在前面的文章中提到,Java為每種基本數據類型都提供了對應的包裝器類型,至于為什么會為每種基本數據類型提供包裝器類型在此不進行闡述,有興趣的朋友可以查閱相關資料。在Java SE5之前,如果要生成一個數值為10的Integer對象,必須這樣進行:

1
Integer i = new Integer(10);

  而在從Java SE5開始就提供了自動裝箱的特性,如果要生成一個數值為10的Integer對象,只需要這樣就可以了:

1
Integer i = 10;

  這個過程中會自動根據數值創建對應的 Integer對象,這就是裝箱。

  那什么是拆箱呢?顧名思義,跟裝箱對應,就是自動將包裝器類型轉換為基本數據類型:

1
2
Integer i = 10;  //裝箱
int n = i;   //拆箱

  簡單一點說,裝箱就是  自動將基本數據類型轉換為包裝器類型;拆箱就是  自動將包裝器類型轉換為基本數據類型。

  下表是基本數據類型對應的包裝器類型:

int(4字節)Integer
byte(1字節)Byte
short(2字節)Short
long(8字節)Long
float(4字節)Float
double(8字節)Double
char(2字節)Character
boolean(未定)Boolean

二.裝箱和拆箱是如何實現的

  上一小節了解裝箱的基本概念之后,這一小節來了解一下裝箱和拆箱是如何實現的。

  我們就以Interger類為例,下面看一段代碼:

1
2
3
4
5
6
7
public class Main {
    public static void main(String[] args) {
         
        Integer i = 10;
        int n = i;
    }
}

  反編譯class文件之后得到如下內容:

  

  從反編譯得到的字節碼內容可以看出,在裝箱的時候自動調用的是Integer的valueOf(int)方法。而在拆箱的時候自動調用的是Integer的intValue方法。

  其他的也類似,比如Double、Character,不相信的朋友可以自己手動嘗試一下。

  因此可以用一句話總結裝箱和拆箱的實現過程:

  裝箱過程是通過調用包裝器的valueOf方法實現的,而拆箱過程是通過調用包裝器的 xxxValue方法實現的。(xxx代表對應的基本數據類型)。

三.面試中相關的問題

  雖然大多數人對裝箱和拆箱的概念都清楚,但是在面試和筆試中遇到了與裝箱和拆箱的問題卻不一定會答得上來。下面列舉一些常見的與裝箱/拆箱有關的面試題。

1.下面這段代碼的輸出結果是什么?

1
2
3
4
5
6
7
8
9
10
11
12
public class Main {
    public static void main(String[] args) {
         
        Integer i1 = 100;
        Integer i2 = 100;
        Integer i3 = 200;
        Integer i4 = 200;
         
        System.out.PRintln(i1==i2);
        System.out.println(i3==i4);
    }
}

  也許有些朋友會說都會輸出false,或者也有朋友會說都會輸出true。但是事實上輸出結果是:

truefalse

   為什么會出現這樣的結果?輸出結果表明i1和i2指向的是同一個對象,而i3和i4指向的是不同的對象。此時只需一看源碼便知究竟,下面這段代碼是Integer的valueOf方法的具體實現:

public static Integer valueOf(int i) {        if(i >= -128 && i <= IntegerCache.high)            return IntegerCache.cache[i + 128];        else            return new Integer(i);    }

  而其中IntegerCache類的實現為:

private static class IntegerCache {
static final int high;
static final Integer cache[];

static {
final int low = -128;

// high value may be configured by property
int h = 127;
if (integerCacheHighPropValue != null) {
// Use Long.decode here to avoid invoking methods that
// require Integer's autoboxing cache to be initialized
int i = Long.decode(integerCacheHighPropValue).intValue();
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - -low);
}
high = h;

cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++);
}

private IntegerCache() {}
}

  從這2段代碼可以看出,在通過valueOf方法創建Integer對象的時候,如果數值在[-128,127]之間,便返回指向IntegerCache.cache中已經存在的對象的引用;否則創建一個新的Integer對象。

  上面的代碼中i1和i2的數值為100,因此會直接從cache中取已經存在的對象,所以i1和i2指向的是同一個對象,而i3和i4則是分別指向不同的對象。

2.下面這段代碼的輸出結果是什么?

1
2
3
4
5
6
7
8
9
10
11
12
public class Main {
    public static void main(String[] args) {
         
        Double i1 = 100.0;
        Double i2 = 100.0;
        Double i3 = 200.0;
        Double i4 = 200.0;
         
        System.out.println(i1==i2);
        System.out.println(i3==i4);
    }
}

  也許有的朋友會認為跟上面一道題目的輸出結果相同,但是事實上卻不是。實際輸出結果為:

falsefalse

  至于具體為什么,讀者可以去查看Double類的valueOf的實現。

  在這里只解釋一下為什么Double類的valueOf方法會采用與Integer類的valueOf方法不同的實現。很簡單:在某個范圍內的整型數值的個數是有限的,而浮點數卻不是。

  注意,Integer、Short、Byte、Character、Long這幾個類的valueOf方法的實現是類似的。

     Double、Float的valueOf方法的實現是類似的。

3.下面這段代碼輸出結果是什么:

1
2
3
4
5
6
7
8
9
10
11
12
public class Main {
    public static void main(String[] args) {
         
        Boolean i1 = false;
        Boolean i2 = false;
        Boolean i3 = true;
        Boolean i4 = true;
         
        System.out.println(i1==i2);
        System.out.println(i3==i4);
    }
}

  輸出結果是:

truetrue

  至于為什么是這個結果,同樣地,看了Boolean類的源碼也會一目了然。下面是Boolean的valueOf方法的具體實現:

public static Boolean valueOf(boolean b) {        return (b ? TRUE : FALSE);    }

  而其中的 TRUE 和FALSE又是什么呢?在Boolean中定義了2個靜態成員屬性:

public static final Boolean TRUE = new Boolean(true);

/**
* The <code>Boolean</code> object corresponding to the primitive
* value <code>false</code>.
*/
public static final Boolean FALSE = new Boolean(false);

  至此,大家應該明白了為何上面輸出的結果都是true了。

4.談談Integer i = new Integer(xxx)和Integer i =xxx;這兩種方式的區別。

  當然,這個題目屬于比較寬泛類型的。但是要點一定要答上,我總結一下主要有以下這兩點區別:

  1)第一種方式不會觸發自動裝箱的過程;而第二種方式會觸發;

  2)在執行效率和資源占用上的區別。第二種方式的執行效率和資源占用在一般性情況下要優于第一種情況(注意這并不是絕對的)。

5.下面程序的輸出結果是什么?

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
public class Main {
    public static void main(String[] args) {
         
        Integer a = 1;
        Integer b = 2;
        Integer c = 3;
        Integer d = 3;
        Integer e = 321;
        Integer f = 321;
        Long g = 3L;
        Long h = 2L;
         
        System.out.println(c==d);
        System.out.println(e==f);
        System.out.println(c==(a+b));
        System.out.println(c.equals(a+b));
        System.out.println(g==(a+b));
        System.out.println(g.equals(a+b));
        System.out.println(g.equals(a+h));
    }
}

  先別看輸出結果,讀者自己想一下這段代碼的輸出結果是什么。這里面需要注意的是:當 "=="運算符的兩個操作數都是 包裝器類型的引用,則是比較指向的是否是同一個對象,而如果其中有一個操作數是表達式(即包含算術運算)則比較的是數值(即會觸發自動拆箱的過程)。另外,對于包裝器類型,equals方法并不會進行類型轉換。明白了這2點之后,上面的輸出結果便一目了然:

true
false
true
true
true
false
true

  第一個和第二個輸出結果沒有什么疑問。第三句由于  a+b包含了算術運算,因此會觸發自動拆箱過程(會調用intValue方法),因此它們比較的是數值是否相等。而對于c.equals(a+b)會先觸發自動拆箱過程,再觸發自動裝箱過程,也就是說a+b,會先各自調用intValue方法,得到了加法運算后的數值之后,便調用Integer.valueOf方法,再進行equals比較。同理對于后面的也是這樣,不過要注意倒數第二個和最后一個輸出的結果(如果數值是int類型的,裝箱過程調用的是Integer.valueOf;如果是long類型的,裝箱調用的Long.valueOf方法)。

如果對上面的具體執行過程有疑問,可以嘗試獲取反編譯的字節碼內容進行查看。

  如果有哪位朋友有補充的內容,歡迎下方留言,不勝感激。

作者:海子
    
出處:http://www.survivalescaperooms.com/dolphin0520/
    
本博客中未標明轉載的文章歸作者海子和博客園共有,歡迎轉載,但未經作者同意必須保留此段聲明,且在文章頁面明顯位置給出原文連接,否則保留追究法律責任的權利。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 普格县| 浦城县| 宿州市| 双江| 东城区| 文山县| 兴义市| 岚皋县| 七台河市| 科技| 泸定县| 山阴县| 巴楚县| 景谷| 疏附县| 桃江县| 黑龙江省| 泰顺县| 方正县| 霍城县| 南漳县| 义乌市| 鄄城县| 泸定县| 上饶县| 英吉沙县| 巩义市| 仙游县| 育儿| 于都县| 邻水| 陆良县| 雷山县| 丘北县| 九江县| 海晏县| 常熟市| 柳林县| 鄂温| 新巴尔虎右旗| 文山县|