Java基本数据类型缓存池解析-源码剖析

抛出问题:new Integer(18) 与 Integer.valueOf(18) 的区别是什么?

  • new Integer(18) 每次都会新建一个对象;
  • Integer.valueOf(18) 会使⽤用缓存池中的对象,多次调用只会取同⼀一个对象的引用
Integer x = new Integer(18);
Integer y = new Integer(18);
System.out.println(x == y);Integer z = Integer.valueOf(18);
Integer k = Integer.valueOf(18);
System.out.println(z == k);Integer m = Integer.valueOf(300);
Integer p = Integer.valueOf(300);
System.out.println(m == p);
来看一下输出的结果吧

除了Float和Double之外,其他的六个包装器类(Byte、Short、Integer、Long、Character、Boolean)都有常量缓存池。

为什么那两个没有呢? 很好思考,缓存这一思想,简单理解,是把常用的准备好。

首先,Float、Double。都是不可穷举,就算缓存也不知道缓存1.0、还是1.1等等。但是Boolean,仅仅只有true&false,还有这些整数类都一样(也不是全缓存,而是都缓存了255)。

  • Byte:-128~127,也就是所有的 byte 值
  • Short:-128~127
  • Long:-128~127
  • Character:\u0000 - \u007F
  • Boolean:true 和 false
  • Integer:-128~127

为了知道为什么,来跟随笔者阅读valueof的源码看看吧(bushi~,还是先看下面的注释吧)。

    @IntrinsicCandidatepublic static Integer valueOf(int i) {if (i >= IntegerCache.low && i <= IntegerCache.high)return IntegerCache.cache[i + (-IntegerCache.low)];return new Integer(i);}

贴出注释:

翻译: 返回一个Integer示例,表示指定的int 值。如果不需要一个新的Integer实例,则通常应优先使用此方法,而不是构造函数Integer(int),因为此方法可能通过缓存频繁请求的值来产生更好的空间和时间性能。此方法将始终缓存-128到127(包括两端)范围内的值,

参数: i 一个 int值

返回值:一个表示Integer的实例 i

自1.5起

从这里发现,我们还需要看看IntegerCache这个静态内部类的源码:

private static class IntegerCache {static final int low = -128;static final int high;static final Integer[] cache;static Integer[] archivedCache;static {// high value may be configured by propertyint h = 127;String integerCacheHighPropValue =VM.getSavedProperty("java.lang.Integer.IntegerCache.high");if (integerCacheHighPropValue != null) {try {h = Math.max(parseInt(integerCacheHighPropValue), 127);// Maximum array size is Integer.MAX_VALUEh = Math.min(h, Integer.MAX_VALUE - (-low) -1);} catch( NumberFormatException nfe) {// If the property cannot be parsed into an int, ignore it.}}high = h;// Load IntegerCache.archivedCache from archive, if possibleCDS.initializeFromArchive(IntegerCache.class);int size = (high - low) + 1;// Use the archived cache if it exists and is large enoughif (archivedCache == null || size > archivedCache.length) {Integer[] c = new Integer[size];int j = low;for(int i = 0; i < c.length; i++) {c[i] = new Integer(j++);}archivedCache = c;}cache = archivedCache;// range [-128, 127] must be interned (JLS7 5.1.7)assert IntegerCache.high >= 127;}private IntegerCache() {}}

同样,先来看看注释:

翻译:

缓存支持JLS要求的自动装箱的对象标识语义,适用于-128和127之间的值(包含)。缓存在首次使用时初始化。缓存的大小可以通过-XX:AutoBoxCacheMax=<size> 选项来控制。在VM初始化期间,java.lang.Integer.IntegerCache.high属性可以设置并保存在jdk.internal.misc的私有系统属性中。VM类。警告:缓存与CDS一起存档,并在运行时从共享*存档中重新加载。归档缓存(Integer[])和Integer对象位于封闭的归档堆区域中。更改实现时应小心,初始化后不应为缓存数组分配新的Integer对象。

说人话:就是通过Integer.valueOf()方法获取整数对象时,会先检查该整数是否存在IntegerCache中,如果在,则返回缓存中的对象,否则创建一个新的对象并缓存起来。

 assert IntegerCache.high >= 127;

这里这个assert,是一个关键字,寓意是断言,为了方便表示程序,并不是发布程序的组成部分。

默认情况下,断言是关闭的,可以在命令行允许java程序时候加入 -ea参数来打开断言。

来看这个代码:

public class Test {public static void main(String[] args) {int high = 126;assert high >= 127;}
}

在这里通过命令行,java -ea参数可以看到报错了。这里我们断言high>=127.当不满足时候,就会报错。实际上不止Java有这种设计,C也有,感兴趣的可以去了解。

在Java中,针对一些基本数据类型,Java会在程序启动时候创建一些常用的对象并缓存在内存中,以提高程序的性能和节省内存开销。这些常用对象缓存在一个固定的范围内,超过这个范围的值会被重新创建新的对象。

使用数据类型缓存池可以有效提高程序的性能和节省内存开销,但需要注意的是,在特定业务下,缓存池可能带来一些问题,比如,缓存池中的对象被不同的线程同时修改,导致数据错误等问题。因此,实际开发中,需要根据具体的业务需求来决定是否需要使用数据类型缓存池。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/905125.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WORD压缩两个免费方法

日常办公和学习中&#xff0c;Word文档常常因为包含大量图片、图表或复杂格式而导致文件体积过大&#xff0c;带来诸多不便&#xff0c;比如 邮件发送受限&#xff1a;许多邮箱附件限制在10-25MB&#xff0c;大文件无法直接发送 存储空间占用&#xff1a;大量文档占用硬盘或云…

罗技无线鼠标的配对方法

罗技鼠标的配对方法&#xff1a; 重新连接鼠标 请按照以下步骤将鼠标与 USB 接收器重新配对。 1.将USB接收器插入计算机。 2.将鼠标关闭电源。 3.按住并持续按住向右按钮&#xff0c;直到操作结束。 4.切换鼠标电源。 5. 单击一次左侧按钮。 6. 单击一次中间按钮。 7.全部松开&…

四、Hadoop 2.X vs 3.X:特性、架构与性能全解析

Hadoop 2.X 与 Hadoop 3.X 深度对比&#xff1a;版本特性、架构与性能剖析 在大数据处理的浪潮中&#xff0c;Hadoop 凭借其分布式存储与计算的强大能力&#xff0c;成为了业界的核心框架之一。随着技术的不断演进&#xff0c;Hadoop 也经历了多个重要版本的迭代。其中&#x…

【React中useReducer钩子详解】

useReducer 是 React 中用于管理复杂状态逻辑的 Hook&#xff0c;它通过 集中式状态更新逻辑 替代 useState&#xff0c;尤其适合处理多值关联状态或依赖前序状态更新的场景。以下是其核心要点&#xff1a; 1. 核心概念 Reducer 模式&#xff1a;灵感来自 JavaScript 的 Array…

【C++】C++函数指针详解与实用技巧

C函数指针详解与实用技巧 在C中&#xff0c;**函数指针&#xff08;Function Pointer&#xff09;**是一种强大而灵活的工具&#xff0c;常用于回调机制、策略模式、事件处理等场景。本文将从概念、语法、常见用法到实战示例&#xff0c;带你全面掌握C函数指针。 &#x1f9e0…

【计算机视觉】基于深度学习的实时情绪检测系统:emotion-detection项目深度解析

基于深度学习的实时情绪检测系统&#xff1a;emotion-detection项目深度解析 1. 项目概述2. 技术原理与模型架构2.1 核心算法1) 数据预处理流程2) 改进型MobileNetV2 2.2 系统架构 3. 实战部署指南3.1 环境配置3.2 数据集准备3.3 模型训练3.4 实时推理 4. 常见问题与解决方案4.…

IC ATE集成电路测试学习——电流测试的原理和方法

电流测试 我们可以通过电流来判断芯片的工作状态时&#xff0c;首先先了解下芯片的电流是如何产生的。 静态电流 理论上&#xff0c;CMOS结构的芯片静态时几乎不耗电 CMOS基本结构&#xff1a;Pmos Nmos 串联当逻辑电平稳定时&#xff1a; ➜ 要么Pmos导通&#xff0c;Nmo…

stm32week15

stm32学习 十一.中断 2.NVIC Nested vectored interrupt controller&#xff0c;嵌套向量中断控制器&#xff0c;属于内核(M3/4/7) 中断向量表&#xff1a;定义一块固定的内存&#xff0c;以4字节对齐&#xff0c;存放各个中断服务函数程序的首地址&#xff0c;中断向量表定…

list类的详细讲解

【本节目标】 1. list的介绍及使用 2. list的深度剖析及模拟实现 3. list与vector的对比 1. list的介绍及使用 1.1 list的介绍 1. list 是可以在常数范围内在任意位置进行插入和删除的序列式容器&#xff0c;并且该容器可以前后双向迭代。 2. list 的底层是双向链表结构&a…

第十节:图像处理基础-图像算术运算 (加法、减法、混合)

引言 在计算机视觉领域,图像算术运算是最基础却至关重要的核心技术。无论是实现简单的图片合成、开发智能监控系统,还是构建复杂的医学影像分析工具,加减运算和混合操作都扮演着关键角色。OpenCV作为最流行的计算机视觉库,提供了完善的图像处理函数集。本文将深入解析三种…

【React 的useState钩子详解】

React 的 useState 钩子详解 useState 是 React 中最基础且最常用的 Hook 之一&#xff0c;它允许你在函数组件中添加和管理状态。 基本语法 const [state, setState] useState(initialState);initialState: 状态的初始值&#xff0c;可以是任何 JavaScript 数据类型state:…

vue 中的数据代理

在 Vue 中&#xff0c;数据代理&#xff08;Data Proxy&#xff09; 是 Vue 实现 MVVM 模式 的关键技术之一。Vue 使用数据代理让你可以通过 this.message 访问 data.message&#xff0c;而不需要写 this.data.message —— 这大大简化了模板和逻辑代码。 我们来深入理解它的本…

基于Python的网络电子书阅读系统

标题:基于Python的网络电子书阅读系统 内容:1.摘要 随着数字化阅读的兴起&#xff0c;网络电子书阅读需求日益增长。本研究旨在开发一个基于Python的网络电子书阅读系统&#xff0c;以满足用户便捷阅读电子书的需求。采用Python的Flask框架搭建Web服务器&#xff0c;结合SQLit…

基于SpringBoot的抽奖系统测试报告

一、编写目的 本报告为抽奖系统测试报告&#xff0c;本项目可用于团体抽奖活动&#xff0c;包括了用户注册&#xff0c;用户登录&#xff0c;修改奖项以及抽奖等功能。 二、项目背景 抽奖系统采用前后端分离的方法来实现&#xff0c;同时使用了数据库来存储相关的数据&…

Apache Flink 与 Flink CDC:概念、联系、区别及版本演进解析

Apache Flink 与 Flink CDC:概念、联系、区别及版本演进解析 在实时数据处理和流式计算领域,Apache Flink 已成为行业标杆。而 Flink CDC(Change Data Capture) 作为其生态中的重要组件,为数据库的实时变更捕获提供了强大的能力。 本文将从以下几个方面进行深入讲解: 什…

单片机-STM32部分:9、定时器

飞书文档https://x509p6c8to.feishu.cn/wiki/A749wx8T0ioqfgkzZKlc9poknUf SMT32F1系列共有8个定时器&#xff1a; 基本定时器&#xff08;TIM6、TIM7&#xff09; 通用定时器&#xff08;TIM2、TIM3、TIM4、TIM5&#xff09; 高级定时器&#xff08;TIM1、TIM8&#xff09…

uniapp-商城-51-后台 商家信息(logo处理)

前面对页面基本进行了梳理和说明&#xff0c;特别是对验证规则进行了阐述&#xff0c;并对自定义规则的兼容性进行了特别补充&#xff0c;应该说是干货满满。不知道有没有小伙伴已经消化了。 下面我们继续前进&#xff0c;说说页面上的logo上传组件&#xff0c;主要就是uni-fil…

ideal创建Springboot项目(Maven,yml)

以下是使用 IntelliJ IDEA 创建基于 Maven 的 Spring Boot 项目并使用 YAML 配置文件的详细步骤&#xff1a; 一、创建 Spring Boot 项目 启动项目创建向导 打开 IntelliJ IDEA&#xff0c;点击“File”->“New”->“Project”。 在弹出的“New Project”窗口中&#…

MATLAB中矩阵和数组的区别

文章目录 前言环境配置1. 数据结构本质2. 运算规则&#xff08;1&#xff09;基本运算&#xff08;2&#xff09;特殊运算 3. 函数与操作4. 高维支持5. 创建方式 前言 在 MATLAB 中&#xff0c;矩阵&#xff08;Matrix&#xff09; 和 数组&#xff08;Array&#xff09; 的概…

iTwin 数据报表(只是简单的原型不代表实现)

大概想法是 前端从schema和class中选中感兴趣的property内容生成ecsql语句传递给后端后端解析ecsql并提供公开接口给各个分析工具&#xff0c;如excel&#xff0c;poewerBI等&#xff08;Odata或者直接选择来自网站&#xff09;再由分析工具做进一步的处 还未想好的点 如何存…