问题描述

代码示意:

public static void main(String[] args) {
        List<Integer> nodes1 = new ArrayList<>();
        nodes1.add(1);
        nodes1.add(-1);
        nodes1.add(-129);
        nodes1.add(-128);
        nodes1.add(-130);
        List<Integer> nodes2 = new ArrayList<>();
        nodes2.add(1);
        nodes2.add(-1);
        nodes2.add(-129);
        nodes2.add(-128);
        nodes2.add(-130);
        for (int i=0; i<5; i++){
            System.out.println(nodes1.get(i)==nodes2.get(i));
        }
    }

输出:

true
true
false
true
false

很明显和常识不一致,原因在哪呢:

原因解释

Integer类只对-128~127之间的对象做了缓存,(Integer)127 == (Integer)127两边装箱后,实际指向堆内存中同一个对象,(Integer)129 == (Integer)129,装箱为引用类型后,没有做缓存,指向堆内存中不同对象,所以比较结果为false。

附录

Java Integer 源码:

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) {
            try {
                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);
            } catch( NumberFormatException nfe) {
                // If the property cannot be parsed into an int, ignore it.
            }
        }
        high = h;

        cache = new Integer[(high - low) + 1];
        int j = low;
        for(int k = 0; k < cache.length; k++)
            cache[k] = new Integer(j++);

        // range [-128, 127] must be interned (JLS7 5.1.7)
        assert IntegerCache.high >= 127;
    }

    private IntegerCache() {}
}