74

我正在使用Sonar使我的代码更清晰,它指出我正在使用new Integer(1)而不是Integer.valueOf(1). 因为它似乎valueOf没有实例化一个新对象,所以对内存更友好。怎么valueOf不能实例化一个新对象?它是如何工作的?这对所有整数都是真的吗?

4

4 回答 4

73

Integer.valueOf-128为 的值实现缓存+127。请参阅 Java 语言规范的最后一段,第 5.1.7 节,它解释了装箱的要求(通常根据.valueOf方法实现)。

http://docs.oracle.com/javase/specs/jls/se7/html/jls-5.html#jls-5.1.7

于 2010-06-04T13:29:54.377 回答
28

JavaDoc

public static Integer valueOf(int i) 返回一个表示指定 int 值的 Integer 实例。如果不需要新的 Integer 实例,则通常应优先使用此方法而不是构造函数 Integer(int),因为此方法可能会通过缓存频繁请求的值来显着提高空间和时间性能。

ValueOf通常用于自动装箱,因此(当用于自动装箱时)缓存至少从 -128 到 127 的值以遵循自动装箱规范。

这是valueOfSun JVM 1.5 的实现。?看看整个类,看看缓存是如何初始化的。

public static Integer valueOf(int i) {
    final int offset = 128;
    if (i >= -128 && i <= 127) { // must cache 
        return IntegerCache.cache[i + offset];
    }
    return new Integer(i);
}
于 2010-06-04T14:06:38.193 回答
2

他们正在推动您使用valueOf()而不是new Integer()让该方法valueOf()为您执行此操作,并缓存该值以防您将来再次获得相同的数字。在这种情况下,方法不会实例化新的整数,但会给你缓存的整数,这将使新整数的“创建”更快且内存友好的过程..

这样,如果您是没有经验的 Java 程序员,您可能会遇到很多问题,因为您会得出这样 的结论Integer.valueOf(342)==Integer.valueOf(342),因为您可能(或可能没有)两个整数的指针相同,并且您可能会以某种方式练习它,比如说,您在 C# 中学习过,因此会不时向您显示错误,您将不知道这些错误是如何以及从哪里来的...

于 2011-10-31T13:55:20.550 回答
2

来自 java.lang.Integer 源代码。整数缓存是可配置的。要配置 Sun 以外的整数缓存大小,我们需要java.lang.Integer.IntegerCache.high根据源代码使用 System 属性。

/**
 * Cache to support the object identity semantics of autoboxing for values between 
 * -128 and 127 (inclusive) as required by JLS.
 *
 * The cache is initialized on first usage. During VM initialization the
 * getAndRemoveCacheProperties method may be used to get and remove any system
 * properites that configure the cache size. At this time, the size of the
 * cache may be controlled by the vm option -XX:AutoBoxCacheMax=<size>.
 */

// value of java.lang.Integer.IntegerCache.high property (obtained during VM init)
private static String integerCacheHighPropValue;

static void getAndRemoveCacheProperties() {
    if (!sun.misc.VM.isBooted()) {
        Properties props = System.getProperties();
        integerCacheHighPropValue =
            (String)props.remove("java.lang.Integer.IntegerCache.high");
        if (integerCacheHighPropValue != null)
            System.setProperties(props);  // remove from system props
    }
}

private static class IntegerCache {
    static final int high;
    static final Integer cache[];

    static {
        final int low = -128;

        // high value may be configured by property
        int h = 127;
        if (integerCacheHighPropValue != null) {
            // Use Long.decode here to avoid invoking methods that
            // require Integer's autoboxing cache to be initialized
            int i = Long.decode(integerCacheHighPropValue).intValue();
            i = Math.max(i, 127);
            // Maximum array size is Integer.MAX_VALUE
            h = Math.min(i, Integer.MAX_VALUE - -low);
        }
        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() {}
}

从 java.lang.Short、java.lang.Byte 和 java.lang.Long 创建一个 127 到 -128 的缓存

private static class LongCache {
    private LongCache() {
    }

    static final Long cache[] = new Long[-(-128) + 127 + 1];

    static {
        for (int i = 0; i < cache.length; i++)
            cache[i] = new Long(i - 128);
    }
}

private static class ShortCache {
    private ShortCache() {
    }

    static final Short cache[] = new Short[-(-128) + 127 + 1];

    static {
        for (int i = 0; i < cache.length; i++)
            cache[i] = new Short((short) (i - 128));
    }
}

private static class ByteCache {
    private ByteCache() {
    }

    static final Byte cache[] = new Byte[-(-128) + 127 + 1];

    static {
        for (int i = 0; i < cache.length; i++)
            cache[i] = new Byte((byte) (i - 128));
    }
}
于 2013-01-12T19:30:04.273 回答