這個(gè)題目需要咱們知道點(diǎn)字節(jié)碼,因?yàn)榇蠹叶贾?,很簡單的東西,那他為啥還問你,那肯定希望你回答出來點(diǎn)內(nèi)在東西:
什么是自動(dòng)裝箱,拆箱
先拋出定義,Java中基礎(chǔ)數(shù)據(jù)類型與它們的包裝類進(jìn)行運(yùn)算時(shí),編譯器會(huì)自動(dòng)幫我們進(jìn)行轉(zhuǎn)換,轉(zhuǎn)換過程對(duì)程序員是透明的,這就是裝箱和拆箱,裝箱和拆箱可以讓我們的代碼更簡潔易懂
Java中基礎(chǔ)數(shù)據(jù)類型與它們對(duì)應(yīng)的包裝類見下表(共8種):
當(dāng)表格中左邊列出的基礎(chǔ)類型與它們的包裝類有如下幾種情況時(shí),編譯器會(huì)自動(dòng)幫我們進(jìn)行裝箱或拆箱.
進(jìn)行 = 賦值操作(裝箱或拆箱)
進(jìn)行+,-,*,/混合運(yùn)算 (拆箱)
進(jìn)行>,<,==比較運(yùn)算(拆箱)
調(diào)用equals進(jìn)行比較(裝箱)
ArrayList,HashMap等集合類 添加基礎(chǔ)類型數(shù)據(jù)時(shí)(裝箱)
我們看一段平常很常見的代碼:
public void testAutoBox() {
List<Float> list = new ArrayList<>();
list.add(1.0f);
float firstElement = list.get(0);
}
list集合存儲(chǔ)的是Float包裝類型,我傳入的是float基礎(chǔ)類型,所以需要進(jìn)行裝箱,而最后的get方法返回的是Float包裝類型,我們賦值給float基礎(chǔ)類型,所以需要進(jìn)行拆箱,很簡單,安排的明明白白
具體自動(dòng)裝箱,拆箱,代碼是如何實(shí)現(xiàn)的
既然編譯器幫我們自動(dòng)進(jìn)行了裝箱,拆箱,那么編譯器到底做了些什么,要搞清楚這些,最簡單直接的方式就是看類經(jīng)過編譯器編譯后的字節(jié)碼,下面是上面一段代碼的字節(jié)碼實(shí)現(xiàn)
public testAutoBox()V
L0
LINENUMBER 15 L0
NEW java/util/ArrayList
DUP
INVOKESPECIAL java/util/ArrayList.<init> ()V
ASTORE 1
L1
LINENUMBER 16 L1
ALOAD 1
FCONST_1
INVOKESTATIC java/lang/Float.valueOf (F)Ljava/lang/Float;
INVOKEINTERFACE java/util/List.add (Ljava/lang/Object;)Z
POP
L2
LINENUMBER 17 L2
ALOAD 1
ICONST_0
INVOKEINTERFACE java/util/List.get (I)Ljava/lang/Object;
CHECKCAST java/lang/Float
INVOKEVIRTUAL java/lang/Float.floatValue ()F
FSTORE 2
L3
LINENUMBER 18 L3
RETURN復(fù)制代碼
· L0,對(duì)應(yīng)我們代碼的第一行,new了一個(gè)ArrayList,并賦值給了1號(hào)引用(就是list)。
· L1,先加載list到棧頂,然后FCONST_1指令就是從常量池加載1.0f浮點(diǎn)數(shù)并壓入棧頂(這一塊知識(shí),見附錄1),然后調(diào)用了Float類的靜態(tài) valueOf方法,進(jìn)行裝箱
· ,然后調(diào)用list的add方法。
· L2,先加載list到棧頂,從常量池獲取0(float,int,long,double等基礎(chǔ)類型初始值都是0),調(diào)用list的get方法,檢查是否能轉(zhuǎn)換,調(diào)用了Float的floatValue方法,進(jìn)行拆箱
· ,存儲(chǔ)得到的浮點(diǎn)數(shù)。
所以結(jié)果很明顯了,以float和Float為例,裝箱就是調(diào)用Float的valueOf方法new一個(gè)Float并賦值,拆箱就是調(diào)用Float對(duì)象的floatValue方法并賦值返回給float。其他基礎(chǔ)類型都是大同小異的,具體可以查看源碼。
##自動(dòng)裝箱、拆箱中的坑 ###面試題中經(jīng)常會(huì)有考點(diǎn)就是考察面試者對(duì)Java中自動(dòng)裝箱、拆箱是否了解透徹,比如下面這一道面試題?
public void testAutoBox2() {
//1
int a = 100;
Integer b = 100;
System.out.println(a == b);
//2
Integer c = 100;
Integer d = 100;
System.out.println(c == d);
//3
c = 200;
d = 200;
System.out.println(c == d);
}
請(qǐng)問執(zhí)行結(jié)果是多少?題目很常見,我們來分析一下:
第1段代碼,基礎(chǔ)類型a與包裝類b進(jìn)行==比較,這時(shí)b會(huì)拆箱,直接比較值,所以會(huì)打印 true
第2段代碼,二個(gè)包裝類型,都被賦值了100,所以根據(jù)我們之前的解析,這時(shí)會(huì)進(jìn)行裝箱,調(diào)用Integer的valueOf方法,生成2個(gè)Integer對(duì)象,引用類型==比較,直接比較對(duì)象指針,這里我們先給出結(jié)論,最后會(huì)分析原因,打印 true
跟上面第2段代碼類似,只不過賦值變成了200,直接說結(jié)論,打印 false
結(jié)果是不是很詭異,我們直接去看Integer類valueOf方法的實(shí)現(xiàn)(JDK8的實(shí)現(xiàn))
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
可以看到,這里的實(shí)現(xiàn)并不是簡單的new Integer,而是用IntegerCache做一個(gè)cache,cache的range是可以配置的
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[];
static {
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
int i = parseInt(integerCacheHighPropValue);
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
} catch( NumberFormatException nfe) {
// If the property cannot be parsed into an int, ignore it.
}
}
high = h;
cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++);
這是IntegerCache靜態(tài)代碼塊中的一段,默認(rèn)Integer cache 的下限是-128,上限默認(rèn)127,可以配置,所以到這里就清楚了,我們上面當(dāng)賦值100給Integer時(shí),剛好在這個(gè)range內(nèi),所以從cache中取對(duì)應(yīng)的Integer并返回,所以二次返回的是同一個(gè)對(duì)象,所以==比較是相等的,當(dāng)賦值200給Integer時(shí),不在cache 的范圍內(nèi),所以會(huì)new Integer并返回,當(dāng)然==比較的結(jié)果是不相等的。
附錄1:JVM字節(jié)碼整型的入棧指令有4個(gè),分別是:
iconst(0~5分別對(duì)應(yīng)iconst_0、iconst_1、iconst_2、iconst_3、iconst_4、iconst_5,-1對(duì)應(yīng)iconst_m1)
bipush (-128~127)
sipush (-32768~32767)
ldc (-2147483648~2147483647)
更多關(guān)于“Java培訓(xùn)”的問題,歡迎咨詢千鋒教育在線名師。千鋒已有十余年的培訓(xùn)經(jīng)驗(yàn),課程大綱更科學(xué)更專業(yè),有針對(duì)零基礎(chǔ)的就業(yè)班,有針對(duì)想提升技術(shù)的好程序員班,高品質(zhì)課程助理你實(shí)現(xiàn)java程序員夢(mèng)想。