很久没有写文章了,今天以一段代码重新开始,

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
Integer a = 127;
Integer b = 127;
int c = 127;
System.out.println(a == b);
System.out.println(a == c);
a = 128;
b = 128;
c = 128;
System.out.println(a == b);
System.out.println(a == c);
a = new Integer(127);
b = new Integer(127);
System.out.println(a == b);

这里还给出了代码截图,可以看到这段代码在IDE中有多处warning提示,包括

Condition ‘a == c’ is always ‘true’

Unnecessary boxing ‘new Integer(127)’

Number objects are compared using ‘==’, not ‘equals()’

实际编码时应该按照提示修改warning。这里直接给出执行结果

true

true

false

true

false

了解自动拆箱的知识后,a == c的结果很好理解,包装类型a与原始类型c进行比较时,自动拆箱,相当于127 == 127以及128 == 128,所以c相关的结果都是true。再看一下最后一个a == b,了解equals与==的区别后,结果是false也好理解,这里还提示我们应该使用equals来比较包装类型a和b。本文主要关注前两个a == b,127时结果是true,128时结果是false。

了解自动装箱的知识后,明白这里Integer a = 127等价于Integer a = Integer.valueOf(127)。那现在就看一下valueOf这个方法做了什么。

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.cache[i + (-IntegerCache.low)];
return new Integer(i);
}

原来valueOfIntegerCache这个类相关,那再看一下IntegerCache这个类做了什么。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[];
static {
// high value may be configured by property
int h = 127;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
int i = parseInt(integerCacheHighPropValue);
i = Math.max(i, 127);
// Maximum array size is Integer.MAX_VALUE
h = Math.min(i, Integer.MAX_VALUE - (-low) -1);
}
high = h;
cache = new Integer[(high - low) + 1];
int j = low;
for(int k = 0; k < cache.length; k++)
cache[k] = new Integer(j++);
}
private IntegerCache() {}
}

IntegerCache中定义了一个Integer数组,初始化时,这个数组预先保存了[-128, high]之间的Integer实例。valueOf方法在自动装箱时会判断需要装箱的数是否在这个区间内,如果在则返回预先保存的实例,不在则新实例化。默认的区间范围是[-128, 127],第一次装箱127时,返回的是预先保存的同一个实例,所以结果为true;第二次装箱128时,返回的是新实例化的对象,所以结果为false。

除了Integer内建了IntegerCache,Byte、Short、Long也内建了相应的类。