java中自动装箱拆箱的陷阱


本人曾经实习面试的时候遇到的一个java自动装箱的陷阱,当时就是不太明白,回来自己琢磨了一下这个问题,在此作为总结,

希望让刚开始接触java的同学以后避免这个坑!

好了废话不多说,直接贴出这段代码:

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));	        
	}
大家看到这段代码不妨自己先琢磨一下答案,这其实就是java自动装箱的一个非常常见的陷阱。在讲解这个问题之前,非常有必要

来说一个java规范:特定的基本类型一定得被box成相同的包装类型。这些对象会被高速缓存以重复使用,并且会被当做一般对象使用。

这些特殊的值是boolean值的true和false、所有的byte值、介于-128至127的short与int值,以及介于\u0000与\u007F之间的任何一个char

大家在理解上面所说的java规范,可能前两条的结果大家已经可以的出来了,由于介于-128至127的int值都会被高速缓存以重复

使用,所以第一个输出的结果为true。相反第二条输出的结果为false。

接下来再来讲解后面四条输出语句,首先我们先来说说(a+b)这个过程,其实这个过程是一个拆箱相加,然后装箱的过程。就是让a、b

先拆箱成int类型相加,然后在装箱成Integer类型去和c比较。这样大家应该猜到第3和第5输出语句都为true。

最后再来说一说第4和第6两条输出语句,如果我们在eclipse中,我们首先可以Ctrl第4条输出语句的equals方法,看看里面的源代码:

 public boolean equals(Object obj) {
        if (obj instanceof Integer) {
            return value == ((Integer)obj).intValue();
        }
        return false;
    }
如果大家理解了(a+b)过程,那么自然就会得到答案,因为传入的参数确实为Integer类型,所以第4条语句的结果为true。接着我们

可以Ctrl第6条输出语句的equals方法:

public boolean equals(Object obj) {
        if (obj instanceof Long) {
            return value == ((Long)obj).longValue();
        }
        return false;
    }
因为传入的参数为Integer类型,所以 if条件判断不成立,返回false。所以最后一条输出语句的结果为false。


最后贴上答案:

true
false
true
true
true
false

以上都是本人自己的理解过,若有不妥之处,请指正!





你可能感兴趣的:(java)