溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Java基礎(chǔ)之Integer使用的注意事項(xiàng)是什么

發(fā)布時(shí)間:2021-06-30 09:43:25 來源:億速云 閱讀:121 作者:小新 欄目:編程語言

這篇文章主要介紹了Java基礎(chǔ)之Integer使用的注意事項(xiàng)是什么,具有一定借鑒價(jià)值,感興趣的朋友可以參考下,希望大家閱讀完這篇文章之后大有收獲,下面讓小編帶著大家一起了解一下。

JAVA中Integer對(duì)象的引用

JAVA中沒有指針一說,但也有引用的概念。這里要說的主要是Integer是不是同一個(gè)對(duì)象。

1、先看一段代碼:

public static void main(String[] args){ 
Integer a1 = 100; 
Integer b1 = a1;//另一種也可以b1=100 
Field field = null; 
try { 
field = a1.getClass().getDeclaredField("value"); 
} catch (NoSuchFieldException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} catch (SecurityException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} 

field.setAccessible(true); 

try { 
field.set(a1, 5000); 
} catch (IllegalArgumentException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} catch (IllegalAccessException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} 
System.out.println("b1="+b1); 

Integer c1 = 100; 
System.out.println("c1="+c1); 

}

結(jié)果:

b1=5000

c1=5000

從上面,首先這里要說明幾個(gè),

1)、對(duì)于Integer來說,-128-127之間的整型已經(jīng)初始化放在IntegerCache中,如果是裝箱的話,就會(huì)從這里面取對(duì)象。

2)、b1=a1到底是數(shù)字賦值還是同一個(gè)對(duì)象?這個(gè)從結(jié)果實(shí)際就可以看出來,b1和a1指向同一個(gè)對(duì)象,而不是同一個(gè)數(shù)值

3)、c1=100,說明對(duì)于-128-127之間的數(shù)值,都是從IntegerCache中獲取的對(duì)象,100對(duì)應(yīng)的Integer對(duì)象被改變后,后續(xù)對(duì)于100的裝箱都被改變。因?yàn)楂@取cache中對(duì)象時(shí)用的是數(shù)組索引,而不是數(shù)值比較獲取的。

不過修改這個(gè)緩存會(huì)比較危險(xiǎn),不介意。誰知道什么jar包或者什么平臺(tái)來個(gè)100的裝箱,但得到結(jié)果又不是100,到時(shí)就崩潰了。

2、通過上面描述,那么如果改成這樣又是什么答案

public static void main(String[] args){ 
Integer a1 = 200; 
Integer b1 = a1; 
Field field = null; 
try { 
field = a1.getClass().getDeclaredField("value"); 
} catch (NoSuchFieldException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} catch (SecurityException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} 

field.setAccessible(true); 

try { 
field.set(a1, 5000); 
} catch (IllegalArgumentException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} catch (IllegalAccessException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} 
System.out.println("b1="+b1); 

Integer c1 = 200; 
System.out.println("c1="+c1); 

}

3、那么再改一下

public static void main(String[] args){ 
Integer a1 = new Integer(100); 
Integer b1 = a1; 
Field field = null; 
try { 
field = a1.getClass().getDeclaredField("value"); 
} catch (NoSuchFieldException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} catch (SecurityException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} 

field.setAccessible(true); 

try { 
field.set(a1, 5000); 
} catch (IllegalArgumentException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} catch (IllegalAccessException e) { 
// TODO Auto-generated catch block 
e.printStackTrace(); 
} 
System.out.println("b1="+b1); 

Integer c1 = 100; 
System.out.println("c1="+c1); 

}

這又是什么答案。對(duì)于new的操作,是不進(jìn)行裝箱的,而是在堆中生成對(duì)象的。

理解了裝箱、緩存、引用就不難理解了??梢宰约涸囋?。

先來點(diǎn)基礎(chǔ)的知識(shí)

   基本類型和包裝類的對(duì)應(yīng) 
  byte  Byte 
  short  Short 
  int  Integer 
  long  Long 
  float  Float 
  double  Double 
  char  Character 
  boolean  Boolean

上述的八中基本數(shù)據(jù)類型的對(duì)應(yīng)關(guān)系只有 int->Integer    char->Character    兩個(gè)變化較大,其余都只是將首字母轉(zhuǎn)換為小寫。
再來了解一下JDK5的新特性:自動(dòng)裝箱和拆箱

自動(dòng)裝箱:把基本類型轉(zhuǎn)換為包裝類類型

自動(dòng)拆箱:把包裝類類型轉(zhuǎn)換為基本類型

public class Demo_Integer { 
 public static void main(String[] args) { 
//JDK1.5之前 
 int a = 100; 
 Integer a1 = new Integer(a); //將基本數(shù)據(jù)類型包裝成對(duì)象,裝箱 
 int b = a1.intValue(); //將對(duì)象轉(zhuǎn)換為基本數(shù)據(jù)類型,拆箱 
//JDK1.5之后 
 int x = 100; 
 Integer x1 = x;  //自動(dòng)裝箱,把基本數(shù)據(jù)類型轉(zhuǎn)換為對(duì)象 
 int y = x1 + x;  //自動(dòng)拆箱,把對(duì)象轉(zhuǎn)換為基本數(shù)據(jù)類型 
 } 
}

注意事項(xiàng)

public class Demo_Integer { 
 public static void main(String[] args) { 
  Integer a = null; 
  int b = a + 100; //自動(dòng)拆箱底層將會(huì)調(diào)用a.intValue(),a為null,自然會(huì)拋出 NullPointerException 
  System.out.println(b); 
 } 
}

面試題

public class Demo_Integer { 
 public static void main(String[] args) { 
  Integer i1 = new Integer(97); 
  Integer i2 = new Integer(97); 
  System.out.println(i1 == i2); 
  System.out.println(i1.equals(i2)); 
  System.out.println("-----------"); 
  
  Integer i3 = new Integer(197); 
  Integer i4 = new Integer(197); 
  System.out.println(i3 == i4); 
  System.out.println(i3.equals(i4)); 
  System.out.println("-----------"); 
 } 
}
Output: 
 false 
 true 
 ----------- 
 false 
 true 
 -----------

原因:  

    new 是在堆內(nèi)存開辟空間的,自然比較地址值(==)都為false.  

    由于Integer重寫了equals方法,所以equals輸出都為true. 

你可能感覺太簡(jiǎn)單了,沒有任何技術(shù)含量,因?yàn)樯厦娴牟皇侵攸c(diǎn),看下面代碼

public class Demo_Integer { 
 public static void main(String[] args) { 
  Integer i1 = 127; 
  Integer i2 = 127; 
  System.out.println(i1 == i2); 
  System.out.println(i1.equals(i2)); 
  System.out.println("-----------"); 
 
  Integer i3 = 128; 
  Integer i4 = 128; 
  System.out.println(i3 == i4); 
  System.out.println(i3.equals(i4)); 
  System.out.println("-----------"); 
 } 
}
Output: 
true 
true 
----------- 
false 
true 
-----------

原因:  

    為什么當(dāng)int大于127就是兩個(gè)對(duì)象,127這個(gè)數(shù)字是不是覺得很熟悉?  

    -128到127是byte的取值范圍,如果在這個(gè)取值范圍內(nèi),自動(dòng)裝箱就不會(huì)創(chuàng)建新對(duì)象了,而從常量池中獲取  

    超過了byte的取值范圍就會(huì)在創(chuàng)建新對(duì)象 

自動(dòng)裝箱其底層會(huì)調(diào)用valueOf()方法,簡(jiǎn)單源碼分析(JDK1.8):

public final class Integer extends Number implements Comparable<Integer> { 
 public static Integer valueOf(int i) { 
  //當(dāng) i >= -128 且 i <= 127 時(shí),會(huì)直接將取緩沖區(qū)中的對(duì)象 
  if (i >= IntegerCache.low && i <= IntegerCache.high) 
   return IntegerCache.cache[i + (-IntegerCache.low)]; 
  return new Integer(i);//超過了byte取值范圍會(huì)在堆內(nèi)存創(chuàng)建 
 } 
 
 //內(nèi)部類充當(dāng)緩沖區(qū) 
 private static class IntegerCache { 
  static final int low = -128; 
  static final int high; 
  static final Integer cache[]; 
 
  static { 
   // high value may be configured by property 
   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++); 
 
   // range [-128, 127] must be interned (JLS7 5.1.7) 
   assert IntegerCache.high >= 127; 
  } 
   private IntegerCache() {} 
 } 
  
}

8種基本類型的包裝類和對(duì)象池

java中基本類型的包裝類的大部分都實(shí)現(xiàn)了常量池技術(shù),這些類是Byte,Short,Integer,Long,Character,Boolean,另外兩種浮點(diǎn)數(shù)類型的包裝類則沒有實(shí)現(xiàn)。另外Byte,Short,Integer,Long,Character這5種整型的包裝類也只是在對(duì)應(yīng)值小于等于127時(shí)才可使用對(duì)象池,也即對(duì)象不負(fù)責(zé)創(chuàng)建和管理大于127的這些類的對(duì)象

擴(kuò)展知識(shí)

在jvm規(guī)范中,每個(gè)類型都有自己的常量池。常量池是某類型所用常量的一個(gè)有序集合,包括直接常量(基本類型,String)和對(duì)其他類型、字段、方法的符號(hào)引用。之所以是符號(hào)引用而不是像c語言那樣,編譯時(shí)直接指定其他類型,是因?yàn)閖ava是動(dòng)態(tài)綁定的,只有在運(yùn)行時(shí)根據(jù)某些規(guī)則才能確定具體依賴的類型實(shí)例,這正是java實(shí)現(xiàn)多態(tài)的基礎(chǔ)。

在JVM中,類從被加載到虛擬機(jī)內(nèi)存中開始,到卸載出內(nèi)存為止,它的整個(gè)生命周期包括:加載、驗(yàn)證、準(zhǔn)備、解析、初始化、使用和卸載7個(gè)階段。而解析階段即是虛擬機(jī)將常量池內(nèi)的符號(hào)引用替換為直接引用的過程。

感謝你能夠認(rèn)真閱讀完這篇文章,希望小編分享的“Java基礎(chǔ)之Integer使用的注意事項(xiàng)是什么”這篇文章對(duì)大家有幫助,同時(shí)也希望大家多多支持億速云,關(guān)注億速云行業(yè)資訊頻道,更多相關(guān)知識(shí)等著你來學(xué)習(xí)!

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI