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