Integer缓存池

前言

前面我们了解了==主要用来比较两个对象类型的内存地址,但在平时写代码时,我们会不会碰到这样的现象:对于对象类型,当两个比较对象为某些值时,为什么使用==结果也为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 的陷阱。

相关推荐
森语林溪3 分钟前
大数据环境搭建从零开始(十七):JDK 17 安装与配置完整指南
java·大数据·开发语言·centos·vmware·软件需求·虚拟机
lsx20240632 分钟前
HTML 音频(Audio)详解
开发语言
woshihonghonga35 分钟前
【动手学深度学习】
开发语言·python
郝开41 分钟前
Spring Boot 2.7.18(最终 2.x 系列版本)1 - 技术选型:连接池技术选型对比;接口文档技术选型对比
java·spring boot·spring
威风的虫1 小时前
ES6 数组方法:告别循环,拥抱函数式编程
开发语言·前端·javascript
码界筑梦坊1 小时前
240-基于Python的医疗疾病数据可视化分析系统
开发语言·python·信息可视化·数据分析·毕业设计·echarts
2301_803554521 小时前
C++ 锁类型大全详解
开发语言·c++
wuwu_q1 小时前
用通俗易懂方式,详细讲讲 Kotlin Flow 中的 map 操作符
android·开发语言·kotlin
曼巴UE51 小时前
UE5 C++ Slate 画曲线
开发语言·c++·ue5
小猪咪piggy1 小时前
【项目】小型支付商城 MVC/DDD
java·jvm·数据库