自动拆装箱:就是8种基本类型与其对应包装器类型可以直接互换!
注意:基本类型的数组,与包装器类型的数组不能互换!
自动拆装箱是JDK5.0的新特性之一,这一特性可以使基本类型,与之对应的包装器类型之间直接转换,例如int的包装器类型是Integer!在JDK5.0之后,你甚至可以把int当作成Integer来使用,把Integer当成int来使用。当然,这不是100%的!
自动拆装箱概述
在JDK5.0之后,Java允许把基本类型与其对应的包装器类型之间自动相互转换。
自动装箱:Integer i = 100,把int类型直接赋值给Integer类型;
自动拆装:int a = new Integer(100),把Integer类型直接赋值给int类型。
自动拆装箱原理
其实自动拆装箱是由编译器完成的!我们写的代码,再由编译器“二次加工”,然后再编译成.class文件!那么编译器是怎么“二次加工”的呢?
Integer i = 100:编译器加工为:Integer i = Integer.valueOf(100);
int a = i:编译器加载为:int a = i.intValue();
这也说明一个道理:JVM并不知道什么是自动拆装箱,JVM还是原来的JVM(JDK1.4之前),只是编译器在JDK5.0时“强大”了!
自动拆装箱演变
大家来看看下面代码:
Integer i = 100;//这是自动装箱
Object o = i;//这是身上转型
上面代码是没有问题的,我们是否可以修改上面代码为:
Object o=100;
ok,这是可以的!通过编译器的处理后上面代码为:
Object o = Integer.valueOf(100);
在来看下面代码:
Object o = Integer.valueOf(100);
int a = o;//编译失败!
上面代码是不行的,因为o不是Integer类型,不能自动拆箱,你需要先把o转换成Integer类型后,才能赋值给int类型。
Object o = Integer.valueOf(100);
int a = (Integer)o;
自动拆箱装箱的误区
来看下面代码:
int[] intArr = {1,2,3};
Integer[] integerArr = intArr;//编译失败
上面代码无论编译通过!因为int[]是引用类型,而不是基本类型。而自动拆装箱只是在基本类型与其对应的包装器类型之间进行转换!也就是说,int[]和Integer[]是两种不同的引用类型。
你可能会认为,把上面代码修改下面的样子即可,这说明需要还是没有理解这一概念。把int[]转换成Integer[],等同与把Integer转换成String类型一样,是不行的!所以下面代码还是编译失败。
int[] intArr = {1,2,3};
Integer[] integerArr = (Integer[])intArr;
如果可以把int当成Integer来对待,那么是否可以:int a = null?当然是不行的,所以希望对自动拆装箱正确理解,它不是万能的。
变态小题目
Integer i1 = 100;
Integer i2 = 100;
boolean b1 = i1 == i2;//结果为true
Integer i3 = 200;
Integer i4 = 200;
boolean b2 = i3 == i4;//结果为false
你可能对上面代码的结果感到费解,那么我们来打开这个疑团。第一步,我们先把上面代码通过编译器的“二次加工”处理一下:
Integer i1 = Integer.valueOf(100);
Integer i2 = Integer.valueOf(100);
boolean b1 = i1 == i2;//结果为true
Integer i3 = Integer.valueOf(200);
Integer i4 = Integer.valueOf(200);
boolean b2 = i3 == i4;//结果为false
这时你应该可以看到,疑团在Integer.valueOf()方法身上。传递给这个方法100时,它返回的Integer对象是同一个对象,而传递给这个方法200时,返回的却是不同的对象。这是我们需要打开Integer的源码(这里就不粘贴Integer的源代码了),查看它的valueOf()方法内容。
Integer类的内部缓存了-128~127之间的256个Integer对象,如果valueOf()方法需要把这个范围之内的整数转换成Integer对象时,valueOf()方法不会去new对象,而是从缓存中直接获取,这就会导致valueOf(100)两次,都是从缓存中获取的同一个Integer对象!如果valueOf()方法收到的参数不在缓存范围之内,那么valueOf()方法会new一个新对象!这就是为什么Integer.valueOf(200)两次返回的对象不同的原因了。