首先来看一段代码:
1 Integer x = new Integer(4); 2 Integer y = 4;
在JDK1.5版本后,以上两行代码都能编译通过,那是因为JDK1.5后加入新特性,自动装箱。
第一句代码是正常的创建对象方法,创建了一个Integer包装类对象。
而第二句中,当左边的Interger类型变量指向右边的int基本类型数据时,右边的基本数据类型会自动装箱成Integer对象,即隐式执行了new Integer(4)。
再来一段代码:
1 Integer x = new Integer(4); 2 x = x + 2;
这两句同样可以编译通过,这同样是因为JDK1.5后加入了自动拆箱的特性。
在第二句中,右边的Integer对象与int基本数据类型数据进行加法运算时,Integer对象会自动拆箱,隐式执行了x.intValue(),将Integer对象转换成int基本数据类型数据,然后x.intValue()+2得出和,最后把和自动装箱成Integer对象,new Integer(x.intValue()+2)。这样左边的Integer类型变量x就成功指向了一个新的Integer对象。
从以上例子可以看出,自动拆装箱的新特性可以简化代码,让我们编写程序的时候更加方便。
但是使用中也有要注意的地方,看下面代码:
1 Integer a = new Integer(100); 2 Integer b = new Integer(100); 3 System.out.println("a==b:"+(a==b)); //结果为false。 4 5 Integer c = 100; 6 Integer d = 100; 7 System.out.println("c==d:"+(c==d)); //结果为true。 8 9 Integer m = 200; 10 Integer n = 200; 11 System.out.println("m==n:"+(m==n)); //结果为false。
从这段代码可以看出,在自动拆装箱的过程中,int基本数据类型的大小,会影响到Integer变量最终是否指向的同一个对象,这是为什么?
这段代码三个比较有以下的区别:
1,如果按照正常方式,创建对象(new)来建立变量的引用,因为不同的变量指向的对象也不同,不同变量之间比较当然就为false。
2,如果Integer变量使用自动装箱的形式,对对象建立的引用,而装箱前的int数值大小又正好在byte的范围内(-128~127),那么,如果多个Integer变量指向同一个int值相同的数据装箱成的Integer对象时,java只会在内存创建第一个Integer对象,其他的变量全部都指向了这个对象,所以这些变量比较时,因为它们都指向同一个对象,结果就为true。
3,如果Integer变量使用自动装箱的形式,对对象建立的引用,而装箱前的int数值大小不在byte的范围内(-128~127),那么,即使int的值相同,java也会在内存中对应变量创建多个对象,当多个变量比较时,它们指向的对象不同,所以结果为false。从结果看,这种情况是跟用new来创建对象引用是一样的。
以上例子都是一些应用的小细节,把细节研究透了,java的道路才会更加的平坦。