说拆装箱之前大家首先要知道Java中8种基本数据类型,它们分别是:
- byte
- char
- short
- int
- long
- float
- double
- boolean
java中的基本数据类型其实只是为了方便程序员编程,java本身作为一门面向对象的语言恨不得不用基本数据类型,但是这样做会对广大程序员不友好,所以这也可以看作是原则向方便的一次妥协。这种妥协只是面向程序员的,而对于计算机Java仍然是纯面向对象的。这也就是说我们平时编代码使用起来非常方便的基本数据类型在被编译器编译成JVM能够执行的机器码时还是被转换成了对象类型。
自动装箱是Java编译器在基本数据类型和对应的对象包装类型之间做的一个转化。反之就是自动拆箱。
java中的基本数据类型(int, float, double...)都对应一种包装类型(Integer, Float, Double...),包装类型是基本数据类型的对象模型,比如:
int i1 = 10;//i1是一个变量,里面保存了数值10
Integer i2 = new Integer();//new了一个包装类型对象,i2是该对象的引用
简单来说:
1.装箱是把基本数据类型转变成其对应的包装类型
2.拆箱是把包装类型转变成其对应的基本数据类型
大家来看一种场景,现在我要在ArrayList中存储一些整型数据,怎么办?我们知道java集合类不像数组可以存储基本类型数据,java集合类只能存储对象(实际是存储对象的引用,不过一般我们都说存储对象)。
List<int> lst = new ArrayList<int>();//这一步就会编译报错,因为集合类不能存储基本数据类型
lst.add(10);
lst.add(23);
那怎么办呢?难道没办法存储了吗?其实不然,既然你集合类不能存储基本数据类型的数据只能存储对象,那我们自然的就会想到能不能把基本数据类型的数据变成对象?答案是可以的,如下:
List<Integer> lst = new ArrayList<Integer>();
int i1 = 10;
int i2 = 23;
Integer wrapper1 = new Integer(i1);//把基本数据类型i1变成一个对象wrapper1(这叫手动装箱)
Integer wrapper2 = Integer.valueof(i2);
lst.add(wrapper1);
lst.add(wrapper2);
在java5.0之前,我们要想在集合类中存储基本数据类型就必须像上面一样进行手动装箱,但是在java5.0之后java支持了自动拆装箱,如下:
List<Integer> lst = new ArrayList<Integer>();
int i1 = 10;
int i2 = 23;
Integer wrapper1 = i1;//对i1进行自动装箱
Integer wrapper2 = i2;//对i2进行自动装箱
lst.add(wrapper1);
lst.add(wrapper2);
int i3 = wrapper1;//自动拆箱
这里可以看到,我们在装箱的时候没必要再把基本数据变量作为参数传递给包装类型的构造方法来new一个包装类型的变量,而是直接把基本数据类型的变脸幅值给一个包装类型的引用。其实这里就是java自动帮我们装箱了。
另外,在运算的时候也可以进行自动拆装箱,如下:
Integer i = 10;//自动装箱
System.out.println(i + 20);//这里是先对i进行拆箱变成基本数据类型,然后再进行加法运算
补充两点:
1.自动装箱实际是调用了Integer.valueof()方法
2.自动装箱其实是一种编译器蜜糖(Compiler sugar,编译器在骗我们,让我们感觉这个api好好用,好简洁,实际上它在编译的时候还是装换成了“笨笨的方式”,好像在拿蜜糖逗我们玩),如下:
Integer i = 100;
//实际Jdk在编译的时候会自动将上面的代码变成如下形式:
Integer i = Integer.valueof(100);