今天看《深入理解JAVA虚拟机》,发现自己对自动装箱和拆箱有一些误解,在此写下来:
问题1、
Integer a = 3;
Integer b = 3;
请问 a==b 返回 true 还是 false ?
开始毫不犹豫选择了false ,明显 a和b不是同一个对象。结果亲自运行这段代码发现返回true。
问题2、
Integer c = 333;
Integer d= 333;
请问 c==d 返回 true 还是 false ?
亲自试验之后问题2返回false。c.equals(d) 返回true。
不解了吧?如果问题1返回true,问题2为什么返回false呢?
首先说下自动装箱:Integer自动装箱时是调用valueOf(int i)来完成自动装箱的,再来看Integer的valueOf(int i)的源码:
public static Integer valueOf(int i) {
final int offset = 128;
if (i >= -128 && i <= 127) { // must cache
return IntegerCache.cache[i + offset];
}
return new Integer(i);
}
原来-128-127之间的整数返回的是缓存,大于这个区间的才返回新的Integer对象。
再来看看Integer的equals方法:
public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}
不难看出Integer对象的equals方法比较的是Integer拆箱之后的值。自动拆箱是调用initValue方法来实现的。
总结:对于基本类型的包装类型,Byte、Short、Integer、Long、Character 都存在这样的缓存机制,大家在使用字面量初始化对象,然后进行== 比较的时候需要留心注意。
Float、Double没有这样的机制。
Boolean 计较特使,字面量的初始化后,== 操作会返回Boolean类静态变量的计较,也可以当做一种缓存吧。