包装类缓存对象

在 Java 中,包装类(如 IntegerLongCharacter 等)为了提高性能和节省内存,对一定范围内的值进行了缓存。这种缓存机制使得在某些情况下,相同的值会返回相同的对象,而不是创建新的对象。

1. 包装类的缓存机制

  • 缓存范围
    • Integer:默认缓存 -128127 之间的值。
    • Long:默认缓存 -128127 之间的值。
    • Character:缓存 0127 之间的值。
    • Byte:缓存所有可能的值(-128127)。
    • Short:默认缓存 -128127 之间的值。
    • Boolean:缓存 truefalse
  • 缓存实现
    • 包装类在内部维护了一个缓存数组(如 IntegerCache),在缓存范围内的值会直接从缓存中返回对象。

2. 示例:Integer 的缓存机制

  • 缓存范围内的值
    • -128127 之间的值会返回相同的对象。
  • 缓存范围外的值
    • 超出缓存范围的值会创建新的对象。

代码示例:

java 复制代码
public class IntegerCacheExample {
    public static void main(String[] args) {
        Integer a = 127; // 使用缓存
        Integer b = 127; // 使用缓存
        System.out.println(a == b); // true

        Integer c = 128; // 超出缓存范围,创建新对象
        Integer d = 128; // 超出缓存范围,创建新对象
        System.out.println(c == d); // false
    }
}

输出:

复制代码
true
false

3. 修改 Integer 的缓存范围

  • 通过 JVM 参数调整缓存范围
    • 可以通过 -XX:AutoBoxCacheMax=<size> 参数调整 Integer 的缓存上限。
  • 示例
    • 将缓存范围扩大到 -128200

运行命令:

复制代码
java -XX:AutoBoxCacheMax=200 IntegerCacheExample

代码示例:

java 复制代码
public class IntegerCacheExample {
    public static void main(String[] args) {
        Integer a = 200; // 使用缓存
        Integer b = 200; // 使用缓存
        System.out.println(a == b); // true
    }
}

输出:

复制代码
true

4. 其他包装类的缓存机制

  • Long 缓存机制
    • 默认缓存 -128127 之间的值。

代码示例:

java 复制代码
public class LongCacheExample {
    public static void main(String[] args) {
        Long a = 127L; // 使用缓存
        Long b = 127L; // 使用缓存
        System.out.println(a == b); // true

        Long c = 128L; // 超出缓存范围,创建新对象
        Long d = 128L; // 超出缓存范围,创建新对象
        System.out.println(c == d); // false
    }
}

输出:

复制代码
true
false
  • Character 缓存机制
    • 缓存 0127 之间的值。

代码示例:

java 复制代码
public class CharacterCacheExample {
    public static void main(String[] args) {
        Character a = 127; // 使用缓存
        Character b = 127; // 使用缓存
        System.out.println(a == b); // true

        Character c = 128; // 超出缓存范围,创建新对象
        Character d = 128; // 超出缓存范围,创建新对象
        System.out.println(c == d); // false
    }
}

输出:

复制代码
true
false

5. 缓存机制的意义

  • 性能优化
    • 缓存机制避免了频繁创建和销毁对象,提高了性能。
  • 节省内存
    • 对于常用的小范围值,直接使用缓存对象可以减少内存占用。
  • 注意事项
    • 在比较包装类对象时,应使用 equals() 方法而不是 ==,因为 == 比较的是对象的引用,而不是值。

代码示例:

java 复制代码
public class WrapperComparison {
    public static void main(String[] args) {
        Integer a = 128;
        Integer b = 128;

        System.out.println(a == b); // false
        System.out.println(a.equals(b)); // true
    }
}

输出:

复制代码
false
true

6. 自定义包装类的缓存

  • 如果需要实现自定义包装类的缓存机制,可以参考 IntegerCache 的实现方式。
  • 示例 :自定义一个 MyInteger 类,并实现缓存机制。

代码示例:

java 复制代码
public class MyInteger {
    private int value;
    private static final MyInteger[] cache = new MyInteger[256];

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

    private MyInteger(int value) {
        this.value = value;
    }

    public static MyInteger valueOf(int value) {
        if (value >= -128 && value <= 127) {
            return cache[value + 128];
        }
        return new MyInteger(value);
    }

    @Override
    public boolean equals(Object obj) {
        if (obj instanceof MyInteger) {
            return this.value == ((MyInteger) obj).value;
        }
        return false;
    }

    @Override
    public String toString() {
        return Integer.toString(value);
    }
}

public class MyIntegerCacheExample {
    public static void main(String[] args) {
        MyInteger a = MyInteger.valueOf(127); // 使用缓存
        MyInteger b = MyInteger.valueOf(127); // 使用缓存
        System.out.println(a == b); // true

        MyInteger c = MyInteger.valueOf(128); // 超出缓存范围,创建新对象
        MyInteger d = MyInteger.valueOf(128); // 超出缓存范围,创建新对象
        System.out.println(c == d); // false
    }
}

输出:

复制代码
true
false

总结

包装类的缓存机制是 Java 中一种重要的性能优化手段,适用于常用的小范围值。在使用包装类时,应注意以下几点:

  1. 了解缓存范围,避免不必要的对象创建。
  2. 在比较包装类对象时,使用 equals() 方法而不是 ==
  3. 可以通过 JVM 参数调整 Integer 的缓存范围。
  4. 如果需要,可以参考包装类的缓存机制实现自定义缓存。
相关推荐
coderSong25682 小时前
Java高级 |【实验八】springboot 使用Websocket
java·spring boot·后端·websocket
Mr_Air_Boy3 小时前
SpringBoot使用dynamic配置多数据源时使用@Transactional事务在非primary的数据源上遇到的问题
java·spring boot·后端
豆沙沙包?3 小时前
2025年- H77-Lc185--45.跳跃游戏II(贪心)--Java版
java·开发语言·游戏
年老体衰按不动键盘4 小时前
快速部署和启动Vue3项目
java·javascript·vue
咖啡啡不加糖4 小时前
Redis大key产生、排查与优化实践
java·数据库·redis·后端·缓存
liuyang-neu4 小时前
java内存模型JMM
java·开发语言
UFIT4 小时前
NoSQL之redis哨兵
java·前端·算法
刘 大 望4 小时前
数据库-联合查询(内连接外连接),子查询,合并查询
java·数据库·sql·mysql
怀旧,5 小时前
【数据结构】6. 时间与空间复杂度
java·数据结构·算法
肥仔哥哥19305 小时前
springCloud2025+springBoot3.5.0+Nacos集成redis从nacos拉配置起服务
redis·缓存·最新boot3集成