自动装拆箱
链接: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再去比