Java--裝箱和拆箱詳解
裝箱
八大基本類型都有一個(gè)與之對(duì)應(yīng)的類:
| 基本類型 | 類 |
|---|---|
| byte | Byte |
| short | Short |
| int | Integer |
| long | Long |
| float | Float |
| double | Double |
| char | Character |
| boolean | Boolean |
除了后兩個(gè)Character和Boolean類是Object派生類外,其余六個(gè)是繼承自Number類。
這些類稱為包裝器(wrapper),一旦構(gòu)造了對(duì)象包裝器,就不允許更改包裝器在其中的值同時(shí),對(duì)象包裝器類還是final修飾,所以也不能定義繼承它們的子類。
有時(shí)候需要將基本類型轉(zhuǎn)換為對(duì)象,比如定義一個(gè)整數(shù)型列表,尖括號(hào)中的類型參數(shù)不允許是基本類型,即不允許寫成ArrayList<int>,這時(shí)就需要用到Integer包裝器類,可以聲明一個(gè)Integer對(duì)象的數(shù)組列表ArrayList<Integer>。
而且為了便于添加int類型的元素到ArrayList<Integer>中,下面語(yǔ)句會(huì)自動(dòng)裝箱
list.add(8);
即自動(dòng)地變換成:
list.add(Integer.valueof(8));
再比如Integer num=8;也是自動(dòng)裝箱,會(huì)轉(zhuǎn)換成Integer num=Integer.valueOf(8);,即將基本類型賦值給相應(yīng)的類時(shí),會(huì)觸發(fā)自動(dòng)裝箱。
但是由于裝箱操作會(huì)創(chuàng)建對(duì)象,頻繁的裝箱操作會(huì)消耗許多內(nèi)存,影響性能,所以應(yīng)該盡量避免裝箱。
拆箱
同樣的,將類轉(zhuǎn)換為對(duì)應(yīng)的基本類型的過程就稱為拆箱,如上面的Integer類型變量num,int num2=num;就會(huì)觸發(fā)自動(dòng)拆箱,自動(dòng)地轉(zhuǎn)換為int num2=num.intValue();。
還有在算術(shù)表達(dá)式中也能夠自動(dòng)地裝箱和拆箱,例如:
Integer n=6; n++; n-=2;
編譯器將自動(dòng)地插入一條對(duì)象拆箱的指令,然后進(jìn)行自增計(jì)算,最后再將結(jié)果裝箱。
注意裝箱和拆箱是編譯器認(rèn)可的,而不是虛擬機(jī),編譯器在生成類的字節(jié)碼時(shí),插入必要的方法調(diào)用,而虛擬機(jī)只是執(zhí)行這些字節(jié)碼。
使用數(shù)值對(duì)象包裝器可以將某些基本方法防止在包裝器中,例如parseInt()方法將一個(gè)數(shù)字字符串轉(zhuǎn)換成數(shù)值,parseInt()是一個(gè)靜態(tài)方法,與這里的Integer類對(duì)象沒有任何關(guān)系,只是Integer類是放置這個(gè)方法的一個(gè)好地方罷了。
而我們的拆箱裝箱無(wú)非是自動(dòng)的調(diào)用了放置在類里面的方法如intValue()和valueOf()等。
==
首先看看Integer.valueOf()函數(shù)的源碼,就知道==的坑了。
public static Integer valueOf(int i) {
return i >= 128 || i < -128 ? new Integer(i) : SMALL_VALUES[i + 128];
}
它會(huì)首先判斷 i i i的大?。喝绻?i > = 128 ∣ ∣ i < − 128 i>=128||i<-128 i>=128∣∣i<−128,就創(chuàng)建一個(gè)Integer對(duì)象,否則執(zhí)行SMALL_VALUES[i + 128],再定位到SMALL_VALUES:
private static final Integer[] SMALL_VALUES = new Integer[256];
它是一個(gè)已經(jīng)創(chuàng)建好的靜態(tài)的Integer數(shù)組對(duì)象,也就是說(shuō) i i i在 [ − 128 , 128 ) [-128,128) [−128,128)的范圍內(nèi)時(shí),不會(huì)創(chuàng)建新的對(duì)象,否則會(huì)創(chuàng)建新的對(duì)象,這也就是裝箱為什么創(chuàng)建對(duì)象,從而消耗內(nèi)存。
(
插播反爬信息)博主CSDN地址:https://wzlodq.blog.csdn.net/
比如以下==判斷:
public static void main(String[] args) {
Integer i1=88;
Integer i2=88;
Integer i3=666;
Integer i4=666;
System.out.println(i1==i2);//true
System.out.println(i3==i4);//false
}
==是判斷兩個(gè)對(duì)象的內(nèi)存地址是不是相等,顯然88在區(qū)間(-128,128)內(nèi),直接指向同一個(gè)創(chuàng)建好的數(shù)組,而666則會(huì)重新創(chuàng)建新對(duì)象。
同樣的boolean、byte、char<128;shot、int介于[-128,127]間時(shí),會(huì)包裝到固定的對(duì)象中,比較結(jié)果一定成立,否則會(huì)創(chuàng)建新的對(duì)象,比較結(jié)果不成立。
這樣我們就能知道,混用時(shí)是自動(dòng)拆箱還是自動(dòng)裝箱了,如:
Integer n=666; int m=666; System.out.println(n==m);//true
如果是n自動(dòng)拆箱,則指向常量池同一地址,則結(jié)果為true;如果是m自動(dòng)裝箱,不在區(qū)間范圍內(nèi),創(chuàng)建新對(duì)象,則結(jié)果為false。答案是n自動(dòng)拆箱。
再如:
Integer x=100; int y=200; Long z=300l; System.out.println(x+y==z);//true System.out.println(z.equals(x+y));//false
如果x、y、z自動(dòng)拆箱則指向常量池同一地址,==結(jié)果true;如果x、y拆箱后裝箱成Long,不在區(qū)間范圍內(nèi),創(chuàng)建新對(duì)象,= =結(jié)果是false。答案是會(huì)拆箱。
那equals為什么輸出false?因?yàn)閑quals除了比較值相同外,還會(huì)比較數(shù)據(jù)類型,顯然兩者拆箱后分別是int和long型,故判斷為false。
null
由于包裝類的引用可以為null,所以自動(dòng)裝箱時(shí)可能會(huì)拋出一個(gè)NullPointerException異常,如:
Integer n=null; int m=n;
另外如果在一個(gè)條件表達(dá)式中混合使用Integer和Double類型,Integer值會(huì)拆箱,提升為Double,再裝箱為Double:
Integer n=6; Double m=8.0; System.out.println(true?n:m); //6.0
總結(jié)
本篇文章就到這里了,希望能給你帶來(lái)幫助,也希望您能夠多多關(guān)注腳本之家的更多內(nèi)容!
相關(guān)文章
Spring Security SecurityContextHolder組件示例說(shuō)明
SpringSecurity的SecurityContextHolder組件是存儲(chǔ)當(dāng)前安全上下文的地方,包括認(rèn)證用戶信息,它支持全局訪問、線程局部存儲(chǔ)和上下文傳播,是SpringSecurity認(rèn)證和授權(quán)的核心,文章通過示例展示了如何訪問已認(rèn)證用戶的詳細(xì)信息、手動(dòng)設(shè)置認(rèn)證信息以及使用認(rèn)證信息保護(hù)方法2024-11-11
深入學(xué)習(xí)Java編程中的字符串的進(jìn)階使用
這篇文章主要介紹了Java編程中的字符串的高級(jí)運(yùn)用,包括StringBuffer類和StringTokenizer類以及常量池的介紹,需要的朋友可以參考下2016-01-01
JavaWeb Servlet生命周期細(xì)枝末節(jié)處深究
Servlet指在服務(wù)器端執(zhí)行的一段Java代碼,可以接收用戶的請(qǐng)求和返回給用戶響應(yīng)結(jié)果,下面這篇文章主要給大家介紹了關(guān)于JavaWeb.servlet生命周期的相關(guān)資料,需要的朋友可以參考下2022-10-10
java集合框架的體系結(jié)構(gòu)詳細(xì)說(shuō)明
最近在一本J2EE的書中看到了很不錯(cuò)的對(duì)集合框架的說(shuō)明文章2012-11-11
關(guān)于Java?中?Future?的?get?方法超時(shí)問題
這篇文章主要介紹了Java?中?Future?的?get?方法超時(shí),最常見的理解就是,“超時(shí)以后,當(dāng)前線程繼續(xù)執(zhí)行,線程池里的對(duì)應(yīng)線程中斷”,真的是這樣嗎?本文給大家詳細(xì)介紹,需要的朋友參考下吧2022-06-06
Spring Boot 與 Kotlin 使用Redis數(shù)據(jù)庫(kù)的配置方法
Redis是目前業(yè)界使用最廣泛的內(nèi)存數(shù)據(jù)存儲(chǔ)。下面通過本文給大家介紹Spring Boot 與 Kotlin 使用Redis數(shù)據(jù)庫(kù)的配置方法,感興趣的朋友一起看看吧2018-01-01
Java那點(diǎn)兒事之Map集合不為人知的秘密有哪些
Map用于保存具有映射關(guān)系的數(shù)據(jù),Map集合里保存著兩組值,一組用于保存Map的key,另一組保存著Map的value,和查字典類似,通過key找到對(duì)應(yīng)的value,通過頁(yè)數(shù)找到對(duì)應(yīng)的信息。用學(xué)生類來(lái)說(shuō),key相當(dāng)于學(xué)號(hào),value對(duì)應(yīng)name,age,sex等信息。用這種對(duì)應(yīng)關(guān)系方便查找2021-10-10
Spring JPA事務(wù)管理與自定義操作實(shí)例解析(最新推薦)
在Spring框架中,數(shù)據(jù)持久化操作常常與事務(wù)管理緊密相關(guān),本文將深入探討Spring Data JPA中的事務(wù)管理機(jī)制,并結(jié)合具體實(shí)例,展示如何自定義事務(wù)行為以滿足不同的業(yè)務(wù)需求,感興趣的朋友一起看看吧2024-12-12
java 對(duì)稱加密算法實(shí)現(xiàn)詳解
這篇文章主要介紹了java 對(duì)稱加密算法實(shí)現(xiàn)詳解,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2019-07-07

