一、对象包装器与自动拆装箱
有时,需要将int这样的基本类型转换为对象。所有的基本类型都有一个与之对应的类。例如,Integer类对应基本类型int。通常,这些类成为包装器(wrapper)。它们分别为:Integer、Float、Long、Double、Short、Byte、Character、Void和Boolean(前6个类派生于公共超类Number)。对象包装器是不可变的,即一旦构造了包装器,就不允许更改包装在其中的值。同时,对象包装类还是 final,因此不能定义它们的子类。
假设想定义一个整形数组列表,尖括号内是不允许写基本数据类型的。不能写ArrayList <int>
。这里就要用到 Integer对象包装器:ArrayList <Integer> list = new ArrayList();
。
装箱:
在Java中,这个调用list.add(3)
将自动变换成list.add(Integer.valueOf(3))
。且由于每个值都包装在对象中,所有 ArrayList的效率远低于 int[ ]数组。因此,应该用它构造小型集合,效率至上。
拆箱:
相反的,当将一个 Integer 对象赋给一个 int 值时,将会自动拆箱。也就是说,编译器将以下语句:int n = list.get(i)
翻译为int n = list.get(i).intValue()
。
二、int、Integer和IntHolder的区别
int
int 是Java的基本数据类型,如 int a = 100是储存在栈(stack)中的。
Integer
Integer是 int 的封装以及对 int 的包装同时提供静态方法。比如:Integer a = new Integer(100);首先a时一个对象并且存放在堆中。看下面代码:
Integer a = new Integer(100);
Integer b = 100; //有个装包的过程
int c = 100;
System.out.println(a == b); //false
System.out.println(a == c); //true a在和基础类型对比的时候有个拆包的过程
那么问题就来了,为什么a == b会返回false呢?我们先来看这段代码:
Integer i1 = 127;
Integer i2 = 127;
System.out.println(i1 == i2); //true
Integer i3 = 128;
Integer i4 = 128;
System.out.println(i3 == i4); //false
很奇怪的结果,为什么上下结果不一样呢?这就设计到源代码中的valueOf()方法,其实我们前面讲到Integer i1 = 127;
被翻译为Integer.valueOf(127)
,所以重点就在于这个valueOf()方法。我们先来看它的源码:
public static Integer valueOf(int i) {
//low为-128, high一开始被定义为127
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
所以我们可以看到,对于-128到127之间的数,会存入缓存。Integer i1=127,会对127进行缓存,下次Integer i2=127的时候,直接返回缓存中的数,不会产生新的对象。
而如果不在这个范围内,返回的就是new Integer,一个新的对象,当然不相等。另外,如果在语句中显示地使用了new,那么即使在-128到127之间也是false。
Integer和int比,无论是否使用new都是true,因为会自动拆箱。
IntHolder
IntHolder持有者类型:因为java中只存在值传递,同时包装类一旦创建其数值是不能改变的。所有不可能实现一个方法改变一个数据类型的值。比如:
public class Test {
public static void main(String[] args){
int a = 3;
triple(a);
System.out.println(a);
}
public static void triple(int x) {
x = 3 * x;
}
}
如果想编写一个修改数据值参数的方法,就需要使用在 org.omg.CORBA 包中定义的持有者(holder)类型,包括IntHolder、BooleanHolder等。每个持有者类型都包含一个共有域值,通过它可以访问储存在其中的值。
public class Test {
public static void main(String[] args){
IntHolder a = new IntHolder();
a.value = 3;
triple(a);
System.out.println(a.value);
}
public static void triple(IntHolder x) {
x.value = 3 * x.value;
}
}