0

0

Java中的包装类为什么需要缓存_IntegerCache源码深度剖析

P粉602998670

P粉602998670

发布时间:2026-03-11 12:42:12

|

823人浏览过

|

来源于php中文网

原创

因为integer在-128到127范围内复用缓存对象,超出范围则新建实例;==比较引用,故127相等而128不等;缓存由integercache实现,范围可调但下界固定为-128。

java中的包装类为什么需要缓存_integercache源码深度剖析

为什么 Integer.valueOf(127) == Integer.valueOf(127)Integer.valueOf(128) != Integer.valueOf(128)

因为 Integer 在 -128 到 127 范围内复用缓存对象,超出范围则每次新建实例。这和 == 比较的是引用而非值直接相关。

核心逻辑在 IntegerCache:JVM 启动时就初始化一个静态数组,缓存了 [-128, 127] 共 256 个 Integer 实例。调用 valueOf(int) 时先查这个数组,命中就返回缓存对象。

  • Integer.valueOf() 是推荐的构造方式,它会走缓存;而 new Integer() 总是新建对象(Java 9+ 已废弃)
  • 缓存范围可由 JVM 参数 -XX:AutoBoxCacheMax=200 扩大(仅对 Integer 有效),但不会缩小下界(-128 固定)
  • 其他包装类也有类似机制:ByteShortCharacter 缓存范围固定且不可调;LongDouble 不缓存(JDK 里没实现 valueOf(long) 的缓存分支)

IntegerCache 在哪?怎么验证它真被用了

源码位置在 java.lang.Integer 内部,是个私有静态类,字段叫 IntegerCache.cache。你不能直接 import 或访问它,但能间接验证。

最简单的验证方式是看对象身份:

Text-To-Song
Text-To-Song

免费的实时语音转换器和调制器

下载

立即学习Java免费学习笔记(深入)”;

System.out.println(Integer.valueOf(127) == Integer.valueOf(127)); // true  
System.out.println(Integer.valueOf(128) == Integer.valueOf(128)); // false  
System.out.println(Integer.valueOf(127).equals(Integer.valueOf(127))); // true(值比较,永远成立)
  • 注意:IDE 可能对字面量自动优化(比如 Integer i = 127; 也会触发 valueOf),所以别只靠赋值语句判断
  • 反编译 javap -c YourClass 能看到字节码里实际调的是 Integer.valueOf(int),不是构造器
  • IntegerCache 初始化发生在类加载阶段,且是懒加载——第一次调用 valueOf 时才真正填充数组(不过通常就是启动时)

缓存带来的性能与陷阱:自动装箱场景下最容易翻车

缓存本意是减少小整数对象分配开销,但它让引用相等性变得“不透明”。尤其在集合操作、并发计数、条件判断中容易误判。

  • Map 键冲突:如果把 Integer 当 key,用 == 判断是否同一 key,-128~127 会意外命中,128+ 就失效
  • Mock 测试失败:用 Mockito.eq(128) 匹配参数时没问题,但若被测代码内部用了 == 做校验,mock 行为可能和真实运行不一致
  • 序列化/反序列化后对象必然不等:哪怕原值是 127,反序列化得到的是新对象,== 一定为 false
  • GC 影响极小:缓存对象生命周期与类加载器绑定,基本不参与 GC,但也不建议人为扩大缓存(比如设成 10000),会白占堆内存

为什么 BooleanCharacter 也缓存,但规则不同

Boolean 只缓存 TRUEFALSE 两个实例,所以 Boolean.valueOf(true) == Boolean.TRUE 永远成立;Character 缓存 \u0000 到 \u007f(即 0~127),和 Integer 上界一致,但下界是 0 不是 -128。

  • Character.valueOf(char) 的缓存逻辑在源码里是硬编码循环填 128 个对象,没有配置项
  • Boolean 的缓存甚至没用数组,就两个 static final 字段,连“缓存类”都不需要
  • 别试图用 new Boolean(true) 绕过缓存——它不进池子,但 JDK 9+ 同样已废弃,且语义上也没必要

缓存行为不是语言规范强制的,而是 OpenJDK 的实现细节。理论上别的 JVM 可以不缓存,但所有主流实现都跟进了——因为太实用。真正要小心的,从来不是“有没有缓存”,而是“有没有意识到它存在”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
java中boolean的用法
java中boolean的用法

在Java中,boolean是一种基本数据类型,它只有两个可能的值:true和false。boolean类型经常用于条件测试,比如进行比较或者检查某个条件是否满足。想了解更多java中boolean的相关内容,可以阅读本专题下面的文章。

366

2023.11.13

java boolean类型
java boolean类型

本专题整合了java中boolean类型相关教程,阅读专题下面的文章了解更多详细内容。

42

2025.11.30

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

1010

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

609

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

334

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

235

2025.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

334

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

105

2025.10.23

C# ASP.NET Core微服务架构与API网关实践
C# ASP.NET Core微服务架构与API网关实践

本专题围绕 C# 在现代后端架构中的微服务实践展开,系统讲解基于 ASP.NET Core 构建可扩展服务体系的核心方法。内容涵盖服务拆分策略、RESTful API 设计、服务间通信、API 网关统一入口管理以及服务治理机制。通过真实项目案例,帮助开发者掌握构建高可用微服务系统的关键技术,提高系统的可扩展性与维护效率。

3

2026.03.11

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11万人学习

Java 教程
Java 教程

共578课时 | 80万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号