黄色网址大全免费-黄色网址你懂得-黄色网址你懂的-黄色网址有那些-免费超爽视频-免费大片黄国产在线观看

專注Java教育14年 全國咨詢/投訴熱線:400-8080-105
動力節點LOGO圖
始于2009,口口相傳的Java黃埔軍校
首頁 學習攻略 Java中什么是裝箱和拆箱及Java裝箱和拆箱是如何實現的

Java中什么是裝箱和拆箱及Java裝箱和拆箱是如何實現的

更新時間:2019-09-07 09:00:00 來源:動力節點 瀏覽2784次

  

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


  一、什么是JAVA裝箱?什么是JAVA拆箱?


  Java為每種基本數據類型都提供了對應的包裝器類型,在Java SE5之前,如果要生成一個數值為10的Integer對象,必須這樣進行:


Integer i = new Integer(10);


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


Integer i = 10;


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


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


Integer i = 10; //裝箱


int n = i; //拆箱


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


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


1.png


  二、JAVA 裝箱和拆箱是如何實現的


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


public class Main {

public static void main(String[] args) {

Integer i = 10;

int n = i;

}

}


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


1567585680226314.png


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


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


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


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



  三、面試中相關的問題


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


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


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。但是事實上輸出結果是:


true


false


 

  為什么會出現這樣的結果?輸出結果表明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、下面這段代碼的輸出結果是什么?


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);

}

}


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


  false


  false



1.png


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


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


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



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


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);

}

}


  輸出結果是:


  true


  true


  至于為什么是這個結果,同樣地,看了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 Boolean object corresponding to the primitive 

* value false

*/

public static final Boolean FALSE = new Boolean(false);


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



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


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


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



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


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方法)。


以上就是動力節點java培訓機構小編介紹的“Java中什么是裝箱和拆箱及Java裝箱和拆箱是如何實現的”的內容,希望對大家有幫助,更多java最新資訊請繼續關注動力節點java培訓機構官網,每天會有精彩內容分享與你。


相關免費視頻教程推薦


java基礎視頻教程下載——裝箱與拆箱:http://m.ilovecolors.com.cn/xiazai/2524.html



提交申請后,顧問老師會電話與您溝通安排學習

免費課程推薦 >>
技術文檔推薦 >>
主站蜘蛛池模板: 亚洲精品视频导航 | 激情综合色综合久久综合 | 高清国产精品久久久久 | 国产高清在线精品一区a | 亚洲欧洲日本天天堂在线观看 | 国产成人ae在线观看网站站 | 国产日产综合 | 污香蕉视频在线观看 | 天天干天天添 | 欧美日本一道高清免费3区 欧美日本一二三区 | 在线观看黄色的网站 | 区二区三区四区免费视频 | 日韩成人免费视频 | 在线a人片免费观看不卡 | 日韩欧美成人乱码一在线 | 日日操天天爽 | 午夜视频在线免费看 | 毛片在线播放视频 | 日韩一级特黄毛片在线看 | 2015小明台湾永久区域免费 | 国产色丁香久久综合 | 成人黄色片网站 | 欧美一级鲁丝片 | 欧美在线视频a | 精品视频一区二区 | 天天se天天cao| 日韩精品www| 亚洲人成网站在线观看青青 | 老司机在线永久免费视频观看 | 99在线观看精品视频 | 欧美视频一区二区 | 国产1卡二卡3卡四卡免费 | 人人爽人人看 | 日韩在线一 | 日韩国产欧美一区二区三区 | 久久久国产亚洲精品 | 性欧美人与zozoxxxx视频 | 国产精品一区二区免费 | 午夜爽爽性刺激一区二区视频 | 狠狠色狠狠色综合网 | 国产va免费精品观看 |