深入剖析Java中的裝箱和拆箱
自動裝箱和拆箱問題是Java中一個老生常談的問題了,今天我們就來一些看一下裝箱和拆箱中的若干問題。本文先講述裝箱和拆箱最基本的東西,再來看一下面試筆試中經常遇到的與裝箱、拆箱相關的問題。
以下是本文的目錄大綱:
一.什麼是裝箱?什麼是拆箱?
二.裝箱和拆箱是如何實現的
三.面試中相關的問題
若有不正之處,請諒解和批評指正,不勝感激。
請尊重作者勞動成果,轉載請標明原文連結:
http://www.cnblogs.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。但是事實上輸出結果是:
為什麼會出現這樣的結果?輸出結果表明i1和i2指向的是同一個物件,而i3和i4指向的是不同的物件。此時只需一看原始碼便知究竟,下面這段程式碼是Integer的valueOf方法的具體實現:
而其中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); } } |
也許有的朋友會認為跟上面一道題目的輸出結果相同,但是事實上卻不是。實際輸出結果為:
至於具體為什麼,讀者可以去檢視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); } } |
輸出結果是:
至於為什麼是這個結果,同樣地,看了Boolean類的原始碼也會一目瞭然。下面是Boolean的valueOf方法的具體實現:
而其中的 TRUE 和FALSE又是什麼呢?在Boolean中定義了2個靜態成員屬性:
至此,大家應該明白了為何上面輸出的結果都是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點之後,上面的輸出結果便一目瞭然:
第一個和第二個輸出結果沒有什麼疑問。第三句由於 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
- 深入理解C#的裝箱和拆箱C#
- Java中的自動裝箱與拆箱Java
- 談談JavaScript中裝箱和拆箱JavaScript
- 如何理解Java中的自動拆箱和自動裝箱?Java
- c#的裝箱和拆箱C#
- Java中的自動裝箱與自動拆箱Java
- Java自動拆箱與裝箱Java
- 深入淺出瞭解“裝箱與拆箱”
- Java自動裝箱/拆箱 - Java那些事兒Java
- 【java】JDK5的新特性→→自動裝箱和拆箱JavaJDK
- java中的內部類和自動拆裝箱Java
- Java 效能筆記:自動裝箱/拆箱Java筆記
- [JAVA] Java物件導向之包裝類,拆箱、裝箱Java物件
- 基礎鞏固、探尋Java裝箱和拆箱的奧妙!Java
- C#之拆箱,裝箱C#
- Java語法糖2:自動裝箱和自動拆箱Java
- java空指標出現的情況:拆箱裝箱Java指標
- 通過原始碼瞭解Java的自動裝箱拆箱原始碼Java
- Effective C#:儘量減少裝箱和拆箱C#
- Java學習之自動裝箱和自動拆箱原始碼分析Java原始碼
- 資料型別及拆箱裝箱資料型別
- Visual C#裝箱與拆箱C#
- java基礎(八) 深入解析常量池與裝拆箱機制Java
- .NET Core CSharp 中級篇 2-1 裝箱與拆箱CSharp
- .NET中的六個重要概念:棧、堆、值型別、引用型別、裝箱和拆箱型別
- Java 效能要點:自動裝箱/ 拆箱 (Autoboxing / Unboxing)Java
- dotnet學習筆記一 - 裝箱拆箱 (轉)筆記
- 【轉】.NET中的六個重要概念:棧、堆、值型別、引用型別、裝箱和拆箱型別
- 一文讀懂什麼是Java中的自動拆裝箱Java
- c#之裝箱和取消裝箱C#
- Integer 自動拆箱封箱
- 夯實Java基礎系列2:Java自動拆裝箱裡隱藏的秘密Java
- Java 自動裝箱效能Java
- 【JS迷你書】基本型別之拆箱操作JS型別
- 原來 JS 還存在這樣的拆箱轉換JS