自动装拆箱

链接:https://www.nowcoder.com/questionTerminal/e35f449be29548c68352ef302965a44e

来源:牛客网

题目:

有如下4条语句:()
1 Integer i01=59;
2 int i02=59;
3 Integer i03=Integer.valueOf(59);
4 Integer i04=new Integer(59);

以下输出结果为false的是:

1 System.out.println(i01==i02);
2 System.out.println(i01==i03);
3 System.out.println(i03==i04);
4 System.out.println(i02==i04);

答案解析:

C
 
这道题我是选错了的,看了楼上victor123的答案觉得又学到新知识了,于是乎就搜了一下,写点自己的理解。
首先常量池这个概念,原来以为只要是一个整型,都会放进到常量池,比如,0,1,12222222等。查找后发现,Byte,Short,Integer,Long,Character这5种整型的包装类也只是在对应值小于等于127并且大于等于-128时才可使用常量池,因为他们至占用一个字节(-128~127);
 
再者Integer.valueOf方法中也有判断,如果传递的整型变量>= -128并且小于127时会返回IntegerCache类中一个静态数组中的某一个对象, 否则会返回一个新的Integer对象,代码如下
1 public static Integer valueOf(int i) {
2         assert IntegerCache.high >= 127;
3         if (i >= IntegerCache.low && i <= IntegerCache.high)
4             return IntegerCache.cache[i + (-IntegerCache.low)];
5         return new Integer(i);
6     }
所以如果你测试如下代码
 1 public static void main(String[] args) {
 2          
 3         Integer a = 127;
 4         Integer b = 127;
 5          
 6         Integer c = 128;
 7         Integer d = 128;
 8          
 9         System.out.println(a == b);
10         System.out.println(c == d);
11     }
结合自动封装、常量池以及Integer.valueOf方法就不难得出,答案时true和false;
再看本题
1 Integer i01=59;
2 int i02=59;
3 Integer i03=Integer.valueOf(59);
4 Integer i04=new Integer(59);
第一行:由于59在-128~127范围之内,所以在自动装箱的时候,会返回IntegerCache[59 - (-128)];
第三行:同第一行
第四行:因为有new关键字,所以在heap中开辟了一块新内存放置值为59的Integer对象。
1 System.out.println(i01==i02);//正确
2 System.out.println(i01==i03);//正确,都指向IntegerCache[59-(-128)]对象
3 System.out.println(i03==i04);//错误,引用指向的对象地址不同
4 System.out.println(i02==i04);//正确

其他解析:

①无论如何,Integer与new Integer不会相等。不会经历拆箱过程,
②两个都是非new出来的Integer,如果数在-128到127之间,则是true,否则为false
    java在编译Integer i2 = 128的时候,被翻译成-> Integer i2 = Integer.valueOf(128);而valueOf()函数会对-128到127之间的数进行缓存
③两个都是new出来的,都为false
④int和integer(无论new否)比,都为true,因为会把Integer自动拆箱为int再去比

 
原文地址:https://www.cnblogs.com/gaoyang666/p/11634165.html