首頁(yè)技術(shù)文章正文

Java培訓(xùn): 包裝類概述

更新時(shí)間:2022-10-17 來源:黑馬程序員 瀏覽量:

IT培訓(xùn)班

  - Java中有8中基本數(shù)據(jù)類型,分別是:

1665971512740_1.jpg

  包裝類就是這8種數(shù)據(jù)類型所對(duì)應(yīng)的引用數(shù)據(jù)類型,分別是:

1665971528418_2.jpg

  - 可能有同學(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。

分享到:
在線咨詢 我要報(bào)名
和我們?cè)诰€交談!