Website Logo. Upload to /source/logo.png ; disable in /source/_includes/logo.html

舞乐 VOLER

舞动我人生

关于IntegerCache

Jan 10, 2017

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

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也内建了相应的类。