Java實現(xiàn)拆箱和裝箱的原理解析
什么是拆箱和裝箱?
拆箱和裝箱,其實是Java 5
引入的一個語法糖,是將 Java的8種基本數(shù)據(jù)類型與其對應(yīng)的包裝類之間自動轉(zhuǎn)換的過程,如下圖:
- 裝箱,將基本數(shù)據(jù)類型轉(zhuǎn)換為對應(yīng)的包裝類對象的過程
- 拆箱,將包裝類對象轉(zhuǎn)換為對應(yīng)的基本數(shù)據(jù)類型的過程
如下代碼:Integer a
加上 int b
能正確的得出結(jié)果3
,其中就包含裝箱和拆箱的過程。
public class UnboxingAndPackingTest { public static void main(String[] args) { Integer a = 1; int b = 2; int sum = a + b; System.out.println(sum);// sum=3 } }
如何實現(xiàn)拆箱和裝箱?
在上述代碼中,我們并沒有手動去執(zhí)行裝箱和拆箱,說明這個過程是編譯器自動完成的,那么,編譯器是如何完成裝箱和拆箱的?為了更好地說明裝箱和拆箱的原理,我們還是延用上面的示例代碼,通過字節(jié)碼層面來進(jìn)行分析。
我們通過 javap -c UnboxingAndPackingTest
指令,對字節(jié)碼進(jìn)行反編譯,如下截圖:
裝箱
將上述裝箱的字節(jié)碼摘出來,如下:
0: iconst_1 1: invokestatic #2 // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer; 4: astore_1
字節(jié)碼的解釋如下:
- iconst_1 將整數(shù)常量1 壓入操作數(shù)棧
- invokestatic #2 調(diào)用
Integer.valueOf(int)
方法,將整數(shù)1 裝箱為Integer
對象 - astore_1 將裝箱后的
Integer
對象存儲到局部變量表的索引1 處(即變量 a)
因此,Integer a = 1
在字節(jié)碼層面是調(diào)用了Integer.valueOf(int)
方法,將整數(shù)1 裝箱為Integer
對象。
拆箱
將上述拆箱的字節(jié)碼摘出來,如下:
7: aload_1 8: invokevirtual #3 // Method java/lang/Integer.intValue:()I 11: iload_2 12: iadd 13: istore_3
字節(jié)碼的解釋如下:
- aload_1 將局部變量表索引1 的
Integer
對象加載到操作數(shù)棧 - invokevirtual #3 調(diào)用了
Integer.intValue()
方法,將Integer
對象拆箱為原始類型int
- iload_2 將局部變量表索引2 的
int
值(即變量 b)加載到操作數(shù)棧 - iadd 將兩個
int
值相加 - istore_3 將相加后的結(jié)果存儲到局部變量表索引3 處(即變量 sum)
通過上述字節(jié)碼的解析,我們可以清楚地看到Java編譯器是如何將裝箱和拆箱操作插入到代碼中的。具體總結(jié)成,裝箱操作通過調(diào)用Integer.valueOf(int)
方法實現(xiàn),而拆箱操作通過調(diào)用Integer.intValue()
方法實現(xiàn)。
Integer.valueof()
上述示例的裝箱操作是通過調(diào)用Integer.valueOf(int)
方法實現(xiàn),因此,我們來看看Integer.valueof()
的源碼是如何實現(xiàn)的,源碼截圖如下:
Integer.valueof()
方法,首先會檢查i
是否命中緩存(-128 到 127),如果i
在這個范圍內(nèi),則直接返回緩存中的Integer
實例,如果i
不在緩存范圍內(nèi),則創(chuàng)建一個新的Integer
實例。
這里特別需要注意,Integer的緩存是-128 到 127,如果對兩個 Integer進(jìn)行比較,一定要特別注意==
和equal()
的使用。
下面以一個示例來演示Integer
類型的==
和equal()
使用:
通過Integer
的例子,我們可以很輕易的聯(lián)想到Long
肯定也存在類似的問題,Long.valueof()
源碼如下:
總結(jié)
本文通過具體示例從字節(jié)碼的角度分析了編譯器是如何實現(xiàn)裝箱和拆箱:
- 裝箱和拆箱是java提供的一個語法糖,更加簡化程序員的使用
- 裝箱,將基本數(shù)據(jù)類型轉(zhuǎn)換為對應(yīng)的包裝類對象的過程
- 拆箱,將包裝類對象轉(zhuǎn)換為對應(yīng)的基本數(shù)據(jù)類型的過程
- 對于8種基本類型對應(yīng)的包裝類的比較,推薦使用
equal()
而不是==
盡管裝箱和拆箱在 Java中是一個很簡單的技術(shù)點,但是通過今天的分析,我們不光知道了底層的原理,同時還分析了Integer.valueof()
的源碼,因此,對于Java 8種基本類型對應(yīng)的包裝類的比較的原理也有一個更多的理解,以及在實際開發(fā)中該如何避坑。
以上就是Java實現(xiàn)拆箱和裝箱的原理解析的詳細(xì)內(nèi)容,更多關(guān)于Java拆箱和裝箱的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
springboot2 jackson實現(xiàn)動態(tài)返回類字段方式
這篇文章主要介紹了springboot2 jackson實現(xiàn)動態(tài)返回類字段方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-08-08Java中字符串與byte數(shù)組之間的相互轉(zhuǎn)換
Java語言中字符串類型和字節(jié)數(shù)組類型相互之間的轉(zhuǎn)換經(jīng)常發(fā)生,網(wǎng)上的分析及代碼也比較多,這篇文章將主要介紹Java中字符串與byte數(shù)組之間的相互轉(zhuǎn)換,有需要的朋友們可以參考借鑒,下面來一起看看吧。2016-10-10如何通過Maven倉庫安裝Spire系列的Java產(chǎn)品
這篇文章主要介紹了如何通過Maven倉庫安裝Spire系列的Java產(chǎn)品,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2019-07-07ssm整合之Spring整合MyBatis框架配置事務(wù)的詳細(xì)教程
這篇文章主要介紹了ssm整合之Spring整合MyBatis框架配置事務(wù),本文通過圖文實例代碼相結(jié)合給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-10-10