一、简介
Java为每种基本数据类型都提供了对应的包装类型,装箱就是自动将基本数据类型转换为包装类型;拆箱就是自动将包装类型转换为基本数据类型。
一般可以通过javap -c 命令可以反编译class文件获取拆箱和装箱是如何实现的。就比如Integer,在装箱的时候自动调用的Integer的valueOf(int)方法.而在拆箱的时候自动调用Integer的intValue方法,其他的类型也类似.
因此可以用一句话总结装箱和拆箱的实现过程:
装箱过程是通过调用包装器的valueOf方法实现的,而拆箱过程是通过调用包装器的 xxxValue方法实现的。(xxx代表对应的基本数据类型)。
二、相关问题
(1)在装箱过程中,即通过valueOf方法创建Integer对象的时候,如果数值在[-128,127]之间,会返回常量池中事先已经创建好的对象;否则创建一个新的Integer对象。
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类的valueOf方法采用了与Integer类的valueOf方法不同的实现。因为在某个范围内的整型数值的个数是有限的,而浮点数却不是。注意,Integer、Short、Byte、Character、Long这几个类的valueOf方法的实现是类似的。Double、Float的valueOf方法的实现是类似的。
Double i1 = 100.0;
Double i2 = 100.0;
Double i3 = 200.0;
Double i4 = 200.0;
System.out.println(i1==i2);//false
System.out.println(i3==i4);//false
Boolean类的valueOf直接返回事先创建好的对象。
Boolean i1 = false;
Boolean i2 = false;
Boolean i3 = true;
Boolean i4 = true;
System.out.println(i1==i2);//true
System.out.println(i3==i4);//true
(2)谈谈Integer i = new Integer(xxx)和Integer i =xxx;这两种方式的区别?
答:第一种方式不会触发自动装箱的过程;而第二种方式会触发;另外,如果数值在[-128,127]之间,第二种方式是不会创建新对象的。
(3)相关问题:
需要注意的是:当 "=="运算符的两个操作数都是包装器类型的引用,则是比较指向的是否是同一个对象。
而如果其中有一个操作数是表达式(即包含算术运算)则比较的是数值(即会触发自动拆箱的过程)。
另外,对于包装器类型,equals方法并不会进行类型转换(即数值相同的Long型和Integer型,通过equals比较,必定返回false)。
Integer a = 1;
Integer b = 2;
Integer c = 3;
Integer d = 3;
Integer e = 321;
Integer f = 321;
Long g = 3L;
Long h = 2L;
System.out.println(c==d); //true
System.out.println(e==f); //false
System.out.println(c==(a+b)); //true,a+b先拆箱,再==运算,同样是拆箱
System.out.println(c.equals(a+b));//true,a+b先拆箱,再将int类型的结果装箱
System.out.println(g==(a+b)); //true,a+b先拆箱,再==运算,同样是拆箱
System.out.println(g.equals(a+b));//false,a+b先拆箱,再将int类型的结果装箱,类型不同,直接返回false
System.out.println(g.equals(a+h));//false,a+b先拆箱,运算之后,触发类型晋升,结果为long类型,再装箱,类型相同,值相同,返回true