更新時(shí)間:2022-10-17 來(lái)源:黑馬程序員 瀏覽量:
- Java中有8中基本數(shù)據(jù)類型,分別是:
包裝類就是這8種數(shù)據(jù)類型所對(duì)應(yīng)的引用數(shù)據(jù)類型,分別是:
- 可能有同學(xué)會(huì)問:Java為什么要給基本數(shù)據(jù)類型提供對(duì)應(yīng)的引用數(shù)據(jù)呢?
- 第一,Java是面向?qū)ο蟮恼Z(yǔ)言,給它們?cè)O(shè)計(jì)對(duì)應(yīng)的引用類型,非常符合萬(wàn)物皆對(duì)象的設(shè)計(jì)理念
- 第二,有類,那么就會(huì)有屬性,會(huì)有方法。那么針對(duì)這些類型的一些數(shù)據(jù)操作可以變得更加簡(jiǎn)單,比如說:
- 如果要根據(jù)字節(jié)數(shù)計(jì)算int的取值范圍就有些麻煩,但是它的包裝類Integer就提供了`Integer.MIN_VALUE`和`Integer.MAX_VALUE`記錄了范圍數(shù)據(jù)
- 第三,Java中集合和泛型作為經(jīng)常使用的對(duì)象,它們只支持引用數(shù)據(jù)類型,比如說,如果需要使用集合存儲(chǔ)int類型整數(shù)。那么直接聲明集合的泛型為int是不可以的,此時(shí)就可以使用它對(duì)應(yīng)的包裝類Integer
- 所以結(jié)論就是:使用包裝類可以方便不同數(shù)據(jù)類型的相關(guān)操作,另外集合也會(huì)經(jīng)常用到包裝類!
2. 自動(dòng)裝箱
- 既然包裝類有很好的使用價(jià)值,那么自然就是要?jiǎng)?chuàng)建它們的對(duì)象去使用。而自動(dòng)裝箱就是一種可以很方便快捷的拿到它們對(duì)象的方式,幾種包裝類的設(shè)計(jì)非常相似,這里就以Integer舉例
Integer num = 23;
3. 自動(dòng)拆箱
- 和自動(dòng)裝箱相反,自動(dòng)拆箱指的就是可以將一個(gè)包裝類對(duì)象直接賦值給其對(duì)應(yīng)的基本數(shù)據(jù)類型變量,例如:
Integer num = 23; int a = num;
- 這種情況就是所謂的自動(dòng)拆箱,其實(shí)底層是調(diào)用Integer包裝類的`intValue()`方法,返回了記錄的數(shù)據(jù)值23。以此類推,如果是Double,底層就是調(diào)用的`doubleValue()`方法獲取數(shù)據(jù)值返回。
4. 常見操作
- 基本數(shù)據(jù)類型轉(zhuǎn)字符串:
- 靜態(tài)方法:toString(基本數(shù)據(jù)類型的數(shù)據(jù)值),例如:`String str = Integer.toString(23);`
- 推薦使用String的靜態(tài)方法:valueOf(基本數(shù)據(jù)類型的數(shù)據(jù)值),例如:`String str = String.valueOf(20);`
- 數(shù)值內(nèi)容的字符串轉(zhuǎn)基本數(shù)據(jù)類型:
- 調(diào)用parseXXX的方法,例如:`int num = Integer.parseInt("66");`
- 注意:如果字符串不是數(shù)值內(nèi)容,而是"a"、"b"、"中"...這樣的非數(shù)值,就會(huì)引發(fā)異常:NumberFormatException
5. 面試題
- 觀察以下代碼,說結(jié)果
Integer a = 127; Integer b = 127; System.out.println(a == b); Integer c = Integer.valueOf(127); System.out.println(a == c); Integer d = new Integer(127); System.out.println(a == d); Integer x = 128; Integer y = 128; System.out.println(x == y);
- 結(jié)果分別是:true,true,false,false
- 原因:
- Integer自動(dòng)裝箱底層會(huì)調(diào)用valueOf()方法,源代碼:
public static Integer valueOf(int i) { if (i >= IntegerCache.low && i <= IntegerCache.high) return IntegerCache.cache[i + (-IntegerCache.low)]; return new Integer(i); }
- 可以看到,會(huì)拿數(shù)據(jù)值和Integer的一個(gè)靜態(tài)內(nèi)部類IntegerCache類的low屬性于high屬性做范圍判斷,其中l(wèi)ow的值是-128,high的值是127
- 也就是說,調(diào)用valueOf()方法,會(huì)判斷數(shù)據(jù)是否在-128~127的范圍內(nèi)。如果在范圍內(nèi),就從靜態(tài)內(nèi)部類IntegerCache的一個(gè)cache數(shù)組屬性中獲取一個(gè)Integer對(duì)象
- 如果不在這個(gè)范圍內(nèi),就是新new一個(gè)Integer對(duì)象
- IntegerCache這個(gè)靜態(tài)內(nèi)部類有一個(gè)靜態(tài)代碼塊:
static { // high value may be configured by property int h = 127; String integerCacheHighPropValue = VM.getSavedProperty("java.lang.Integer.IntegerCache.high"); if (integerCacheHighPropValue != null) { try { h = Math.max(parseInt(integerCacheHighPropValue), 127); // Maximum array size is Integer.MAX_VALUE h = Math.min(h, Integer.MAX_VALUE - (-low) -1); } catch( NumberFormatException nfe) { // If the property cannot be parsed into an int, ignore it. } } high = h; // Load IntegerCache.archivedCache from archive, if possible CDS.initializeFromArchive(IntegerCache.class); int size = (high - low) + 1; // Use the archived cache if it exists and is large enough if (archivedCache == null || size > archivedCache.length) { Integer[] c = new Integer[size]; int j = low; for(int i = 0; i < c.length; i++) { c[i] = new Integer(j++); } archivedCache = c; } cache = archivedCache; // range [-128, 127] must be interned (JLS7 5.1.7) assert IntegerCache.high >= 127; }
- 一般情況下,這段代碼第一個(gè)if不會(huì)進(jìn)入,所以high的值就被賦值127,經(jīng)過計(jì)算,size變量的值就是256。
- 第二個(gè)if語(yǔ)句的條件通常是可以成立的,所以就創(chuàng)建了一個(gè)長(zhǎng)度為256的Integer類型數(shù)組,通過一個(gè)for循環(huán),給這個(gè)數(shù)組就從-128開始賦值,一致賦值到127結(jié)束,剛好是256個(gè)。
- 至此,內(nèi)部類中就出現(xiàn)了一個(gè)Integer類型數(shù)組,緩沖了256個(gè)Integer對(duì)象,對(duì)應(yīng)的數(shù)據(jù)范圍正好是:-128~127。
- 那么通過以上分析可知,只要是通過自動(dòng)裝箱或者valueOf()方法去獲取對(duì)象,只要數(shù)據(jù)范圍在-128~127,不管獲取多少次,都會(huì)從數(shù)組中去拿緩沖的對(duì)象,所以拿到的始終是同一個(gè),所以判斷的結(jié)果就是true。
- 但如果不在這個(gè)范圍內(nèi),就是去new一個(gè)新的Integer對(duì)象,會(huì)開辟一個(gè)新的對(duì)象空間,地址值肯定不一樣,所以結(jié)果就是false。