jdk5.0中引入了自动装箱/拆箱(Autoboxing/Unboxing)功能,可以让我们方便地在简单类型和对应的封装类型数据之间转换,例如:
Interger iObject=100;
int i=new Interger(100);
这在JDK5.0之前是非法的。
在JDK5.0中,使用自动装箱的时候还有一个问题需要特别注意:
在java中,为了节省创建对象的时间和空间,对于一些常用的对象,会将它杂内存中缓存起来。String对象就是这样,当直接使用(String s=“str” )这种形式来产生String对象时,如果在内存中已经有一个使用这种方式产生的字符串对象,那么就不会再新建对象,而是直接使用已经存在的那个String对象。例如:
public class ClassA {
/**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
String s1="str";
String s2="str";
System.out.println("s1==s2 ? "+(s1==s2));
System.out.println(s1.hashCode()+" "+s2.hashCode());
String s3=new String("str");
String s4=new String("str");
System.out.println("s3==s4 ? "+(s3==s4));
System.out.println("s1==s4 ? "+(s1==s4));
}
}
输出:
s1==s2 ? true
114225 114225
s3==s4 ? false
s1==s4 ? false
对于如下范围内的简单数据类型:
-
boolean类型的值
-
所有byte类型的值
-
在-128~127之间的short类型的值
-
在-128~127之间的int类型的值
-
在/u000~/u007F之间的char类型的值
它们在使用自动装箱转换成相关封装类型对象的时候,其行为也和String型类似。在上面列表范围中的数据在进行自动装箱的时候,将首先检查内存中是否已经有使用自动装箱产生的具有相同值的对象。如果已经有一个“值”相同的对象存在,那么并不会产生新的对象。这个机制和使用String s=“test”这种方式产生一个字符串对象类似。也就是说,当简单类型的数据列表中的数据类型和对应范围内的值的时候,使用自动装箱得到的对象可能是已经在内存中存在的,而不是新产生的,就跟String类型数据一样。例子:
public class ClassB {
/**
* @param args
*/
public static void main(String[] args) {
// TODO Auto-generated method stub
Integer t1=new Integer(127);
Integer t2=new Integer(127);
System.out.println("t1==t2 ? "+(t1==t2));
Integer t3=127;
Integer t4=127;
System.out.println("t3==t4 ? "+(t3==t4));
System.out.println("t1==t3 ? "+(t1==t3));
Integer t5=128;
Integer t6=128;
System.out.println("t5==t6 ? "+(t5==t6));
Double d1=127.0;
Double d2=127.0;
System.out.println("d1==d2 ? "+(d1==d2));
}
}
输出:
t1==t2 ? false
t3==t4 ? true
t1==t3 ? false
t5==t6 ? false
d1==d2 ? false