1、什么是裝箱?什么是拆箱?
裝箱:基本類型轉變為包裝器類型的過程。
拆箱:包裝器類型轉變為基本類型的過程。
//JDK1.5之前是不支持自動裝箱和自動拆箱的,定義Integer對象,必須 Integer i = new Integer(8); //JDK1.5開始,提供了自動裝箱的功能,定義Integer對象可以這樣 Integer i = 8; int n = i;//自動拆箱
2、裝箱和拆箱的執行過程?
- 裝箱是通過調用包裝器類的 valueOf 方法實現的
- 拆箱是通過調用包裝器類的 xxxValue 方法實現的,xxx代表對應的基本數據類型。
- 如int裝箱的時候自動調用Integer的valueOf(int)方法;Integer拆箱的時候自動調用Integer的intValue方法。
3、常見問題?
- 整型的包裝類 valueOf 方法返回對象時,在常用的取值范圍內,會返回緩存對象。
- 浮點型的包裝類 valueOf 方法返回新的對象。
- 布爾型的包裝類 valueOf 方法 Boolean類的靜態常量 TRUE | FALSE。
實驗代碼
Integer i1 = 100; Integer i2 = 100; Integer i3 = 200; Integer i4 = 200; System.out.println(i1 == i2);//true System.out.println(i3 == i4);//false Double d1 = 100.0; Double d2 = 100.0; Double d3 = 200.0; Double d4 = 200.0; System.out.println(d1 == d2);//false System.out.println(d3 == d4);//false Boolean b1 = false; Boolean b2 = false; Boolean b3 = true; Boolean b4 = true; System.out.println(b1 == b2);//true System.out.println(b3 == b4);//true
- 包含算術運算會觸發自動拆箱。
- 存在大量自動裝箱的過程,如果裝箱返回的包裝對象不是從緩存中獲取,會創建很多新的對象,比較消耗內存。
Integer s1 = 0; long t1 = System.currentTimeMillis(); for(int i = 0; i <1000 * 10000; i++){ s1 += i; } long t2 = System.currentTimeMillis(); System.out.println("使用Integer,遞增相加耗時:" + (t2 - t1));//使用Integer,遞增相加耗時:68 int s2 = 0; long t3 = System.currentTimeMillis(); for(int i = 0; i <1000 * 10000; i++){ s2 += i; } long t4 = System.currentTimeMillis(); System.out.println("使用Integer,遞增相加耗時:" + (t4 - t3));//使用int,遞增相加耗時:6
ps:可深入研究一下 javap 命令,看下自動拆箱、裝箱后的class文件組成。
看一下 JDK 中 Byte、Short、Character、Integer、Long、Boolean、Float、Double的 valueOf 和 xxxValue 方法的源碼(xxx代表基本類型如intValue)。
