前言
前面我们了解了==主要用来比较两个对象类型的内存地址,但在平时写代码时,我们会不会碰到这样的现象:对于对象类型,当两个比较对象为某些值时,为什么使用==结果也为true呢?其实这都是Integer缓存池的功劳。下面我们将介绍这个强大的缓存池。
一、Integer缓存池是什么?
首先,我们先来看一小段代码------
代码示例:
java
public class IntegerCacheDemo {
public static void main(String[] args) {
Integer a = 127;
Integer b = 127;
System.out.println(a == b);
Integer c = 128;
Integer d = 128;
System.out.println(c == d);
}
}
运行结果:

如图,我们创建对照组1和对照组2,两个对照组均使用Integer创建对象,由==进行比较,即比较两者的内存地址。在对照组1中,a与b 的值均为127,对照组2中,c与d 的值均为128。由运行结果我们可知,两个对照组中,对照组1的比较结果为true,表明a,b所指向的地址相同;但在对照组2中,结果表现为false,即c,d所指向的地址不同。造成这种情况的源头就是Integer缓存池。
二、Integer缓存池的底层原理
在JDK中,当我们写下Integer a = 127;时,实际上发生了自动装箱。java编译器会将其转换为:
java
Integer a = Integer.valueOf(127);
重点在于Integer.valueOf(inr i)方法。我们一起来看看源码:
java
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
由源代码可知:
如果传入的整数i在缓存池的范围内,则直接从IntegerCache.cache数组中返回缓存对象。否则就建立一个新的Integer对象。
这就可以解释为什么127时a==b为true,而128时c==d为false。即127为Integer缓存池可保存的最大数据。
三、IntegerCache内部类源码
那么IntegerCache到底是怎么实现的呢?我们继续看源码:
java
private static class IntegerCache {
static final int low = -128;
static final int high;
static final Integer cache[];
static {
int h = 127;
String propValue = sun.misc.VM.getSavedProperty(
"java.lang.Integer.IntegerCache.high");
if (propValue != null) {
int i = Integer.parseInt(propValue);
i = Math.max(i, 127);
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++);
}
}
从源码我们可知:
1.默认缓存范围是[-128,127]。
2.这个上限可以通过JVVM参数-Djava.lang.Integer.IntegerCache.high=xxx来修改。
3.缓存池在类加载时就会被初始化,里面预先放好了从-128到127的Integer对象。
四、验证缓存池的效果
我们来写一段测试代码:
java
public class IntegerCacheTest {
public static void main(String[] args) {
Integer x = Integer.valueOf(127);
Integer y = Integer.valueOf(127);
System.out.println(x == y); // true
Integer m = Integer.valueOf(200);
Integer n = Integer.valueOf(200);
System.out.println(m == n); // false
}
}
如果我们在启动时加上参数:
java
java -Djava.lang.Integer.IntegerCache.high=200 IntegerCacheTest
结果将为:
java
true
true
这是因为缓存池的上限被扩大到了200。
五、Integer缓存池的作用
1.性能优化:Integer时不可变对象,如果每次都新建会浪费内存和增加GC压力。
2.节省空间:小整数在程序中使用非常频繁,例如循环、数组下标、计数器等。缓存池能复用这些对象。
3.提高运行效率:避免了频繁的对象创建与销毁。
六、注意事项
1.不要用==比较数值
==比较的是地址是否相同,在缓存范围内,==可能是true,但范围之外一定是false。正确的做法是equals()。
2.警惕自动装箱陷阱
java
Integer a = 127; // 自动装箱,相当于 Integer.valueOf(127)
Integer b = 127;
System.out.println(a == b); // true
Integer c = 128; // 自动装箱,相当于 Integer.valueOf(128)
Integer d = 128;
System.out.println(c == d); // false
总结
Integer 缓存池是 Java 中一个经典的性能优化机制,它让小整数对象得以复用,避免频繁创建。但同时也容易让人掉进 == 和 equals 的陷阱。