谈谈装箱与拆箱
一、何为包装类型
Java是一种面向对象的语言,但是它不是纯面向对象的。Java中存在基本数据类型,谈不上对象。为了向纯面向对象靠拢,Java5的时候推出了基本数据类型的包装类型。
基本数据类型与包装类型的对应关系如下:
二、何为装箱与拆箱
装箱就是将基本数据类型转化为包装类型,那么拆箱就是将包装类型转化为基本数据类型。
以基本数据类型int为例:
package day1119;
public class TestBox {
public static void main(String[] args) {
//自动装箱,底层其实执行了Integer a=Integer.valueOf(1);
Integer a = 1;
//自动拆箱,底层其实执行了int b=a.intValue();
int b = a;
}
}
Integer的valueOf(int i)方法可以将一个基本数据类型转化为对应的包装类型,即装箱方法。
而Integer的intValue()方法则可以将一个包装类型转化为对应的基本数据类型,即拆箱方法。
三、装箱与自动装箱的区别
装箱:利用Integer的构造方法Integer(int value),即Integer c=new Integer(1);
自动装箱:或者叫隐式装箱,直接给Integer赋值,即Integer d=1,在编译的时候,会调用Integer.valueOf()方法完成装箱。
相比而言,自动装箱可能比装箱具有更高的效率,体现在自动装箱的缓存上,下面从几道题目来讲自动装箱的缓存。
四、相关面试题目
第一题:以下代码的输出结果为?(==号两边如果都是引用类型的话,则判断它们是否指向同一个对象。如果都是基本数据类型的话,则判断它们的数值是否相等)
package day1119;
public class TestBox2 {
public static void main(String[] args) {
Integer a = 100;
Integer b = 100;
Integer c = 200;
Integer d = 200;
System.out.println(a == b);
System.out.println(c == d);
}
}
也许有些人认为他们是四个各不相同的对象,两个式子都返回false。
实际运行后发现输出:
为什么一个是true,一个是false呢?
刚才我们知道,Integer a=100这条语句会触发自动装箱,而自动装箱的方法为Integer.valueOf()方法,让我们去寻找这个方法,一探究竟。
观察Integer类的源码中的valueOf()
/**
* Returns an {@code Integer} instance representing the specified
* {@code int} value. If a new {@code Integer} instance is not
* required, this method should generally be used in preference to
* the constructor {@link #Integer(int)}, as this method is likely
* to yield significantly better space and time performance by
* caching frequently requested values.
*
* This method will always cache values in the range -128 to 127,
* inclusive, and may cache other values outside of this range.
*
* @param i an {@code int} value.
* @return an {@code Integer} instance representing {@code i}.
* @since 1.5
*/
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
可以看得出,当i的值位于[-128,127]的时候,会直接返回Integer缓存数组中相应对象的引用,如果i大于127或小于-128,会重新创建一个Integer实例,并返回。
那么第一条式子a和b的值都在缓存范围内,因此他们指向同一个对象,因此返回true。c和d的值不在范围内,都是通过new创建出来的,因此不是同一个对象,返回false。
注意:Byte、Short、Integer、Long、Character的valueOf()实现机制类似。
其中相同值的Byte比较永远返回true,因为byte取值范围就是[-128,127]。
Short、Integer、Long的valueOf()基本一样,i的范围都需要在[-128,127]。
Character中i的范围只要小于等于127即可,因为char最小值为0,本来就大于等于-128。
但是Float、Double中的valueOf(),永远返回新创建的对象,因为一个范围内的整数是有限的,但是小数却是无限的,无法保存在缓存中。
Boolean中只有两个对象,要么是true的Boolean,要么是false的Boolean,只要boolean的值相同,Boolean就相等。
第二题:以下代码的输出结果为?
package day1119;
public class TestBox3 {
public static void main(String[] args) {
Integer a = 1;
Integer b = 2;
Integer c = 3;
Long d = 2L;
Long e = 3L;
int f = 2;
//一旦有包装类型和数值类型判断==时,则触发包装类型的自动拆箱,转为数值类型的比较
System.out.println(new Integer(300) == 300);//返回true
//一旦有包装类型和数值类型发生运算时,则触发包装类型的自动拆箱,转为数值类型的运算
System.out.println(c == (a + f));//返回true
//一旦有包装类型和包装类型发生运算时,则触发包装类型的自动拆箱,转为数值类型的运算
System.out.println(c == (a + b));//返回true
//只有对象类型才有equals方法,因此首先a,b触发包装类型的自动拆箱,转为数值类型的运算。
//运算完,再将结果3自动装箱,Integer重写了equals,因此可以转为包装类型与包装类型的比较。
//当两边的包装类型不一致时,必定返回false。
//当两边的包装类型一致时,再进行拆箱,判断两者代表的数值是否相等。
System.out.println(c.equals(a + b));//返回true
//不同数据类型的数值进行运算,首先会将低精度的数据类型转化为高精度的数据类型,即自动类型转换。
//比如现在的int+long,会提升到long+long,再进行运算。
System.out.println(e == (a + d));//返回true
//==号两边类型不一致时,直接执行自动拆箱,比较之后的数值
System.out.println(e == (a + b));//返回true
//依次经历自动拆箱,自动类型转换、运算、自动装箱,类型比较,拆箱,数值比较
System.out.println(e.equals(a + d));//返回true
//依次经历自动拆箱,自动类型转换、运算、自动装箱,类型比较,两边类型不一致,直接返回false
System.out.println(c.equals(a + d));//返回false
}
}
五、总结
如果想要深入了解自动装箱拆箱的过程,必须得反编译class文件,了解底层编译的细节,才可以解除自己此方面的疑问。