您的位置:首页 > 编程语言 > Java开发

java基本类型自动装箱与拆箱

2011-12-05 00:10 429 查看
《深入理解JAVA虚拟机-JVM高级特性与最佳实践》的P274有如下代码片段
public static void main(String[] args) {
Integer a =1;
Integer b =2;
Integer c =3;
Integer d =3;
Integer e =321;
Integer f =321;
Long g = 3L;
System.out.println(c==d);
System.out.println(e == f);
System.out.println(c == (a + b));
System.out.println(c.equals(a + b));
System.out.println(g==(a+b));
System.out.println(g.equals(a+b));
}
该代码片段的输出为:true  false true  true  true false

主要原因:

1、包装类自动装箱会缓存值为-128~127的对象

2、包装类“==”运算在没有遇到算术运算的情况下不会自动拆箱(来自该本书)

3、equal的比较会先判断类型,再判断内容

Integer a =1;
的字节码如下:

0:   iconst_1
1:   invokestatic    #16; //Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
4:   astore_1
字节码中的引用#16的valueOf方法的实现代码如下:

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);
}

也就是说自动装箱使用的是valueOf方法,因此Integer a = 1的实际代码为 Integer a = Integer.valueOf(1)。

c==d,c和d引用同1对象,所以为true

e==f,e和f的值超过127,所以值虽一样,但引用的对象不同,所以为false

c==(a+b) ,在执行过程中进行拆箱,最后比较的结果为 c与a+b的值,所以为true

g==(a+b),在执行过程中进行拆箱,计算a+b的值,将使用I2L指令将计算结果3转换成long,然后再跟g的值比较,所以结果为true

a+b的计算结果如果进行自动装箱的话,结果是值为3Integer对象,equal进行比较的时候会使用instanceof先判断类型是否相同,相同再进一步比较,所以g.equal(a+b)的结果为false
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
相关文章推荐