Java源码系列之Integer、Long

简介: Integer、Long 源码解析

1 前言

我们都知道,java提供了8种基本数据类型供我们使用,此外,java还提供了这些基本数据类型所对应的包装类,而且通过自动装箱和自动拆箱机制能够轻松地完成基本数据类型和包装类之间的转换。

  • 自动装箱:自动将基本数据类型转换为包装类。
  • 自动拆箱:自动将包装类转换为基本数据类型。

2 Integer

2.1 总体结构

首先,我们来看一下Integer类的总体结构,如下图所示:


Integer类图结构.jpg

Integer类图结构

  • Integer继承了Number类,并重写了Number类intValue()、longValue()、floatValue()等方法来完成对一些基本数据类型的转换
  • Integer类实现了Comparable接口,这使得我们可以重写compareTo方法来自定义Integer对象之间的比较操作

2.2 注释

从Integer类的注释中我们可以获取到以下信息:

  • Integer类将原始类型int的值包装在一个对象中,Integer类的对象使用一个int类型的字段value来表示对象的值
  • 此外,Integer类还提供了一系列的方法来完成int到String,int到其他基本数据类型,String到int的转换

和String类一样,Integer类也是不可变类,Integer类使用final进行修饰,而且用于表示Integer对象值的字段value也使用了final进行修饰,Java中的所有包装类都是不可变类。

2.3 自动装箱、拆箱原理

通过以下一段简单的代码我们就能了解到自动装箱、拆箱的过程:

publicclassIntegerDemo {
publicstaticvoidmain(String[] args) {
// 自动装箱IntegernumberI=66;
// 自动拆箱intnumber=numberI;
    }
}

那么包装类的自动装箱、拆箱究竟是如何实现的呢?

我们可以通过javap反编译或者通过idea中的插件jclasslib来了解这一过程,如下图所示:


image.jpg


可以看到,Integer类是通过调用自身内部的valueOf()方法来实现自动装箱的,而自动拆箱则是通过调用继承自Number类的intValue()方法来实现的。

2.4 缓存

在《阿里巴巴 Java 开发手册》中有一段关于包装对象之间值的比较问题的规约:就是所有整型包装类对象之间值的比较,全部使用 equals 方法比较。而手册中对这条规约的说明就是:对于 Integer var = ? 在 - 128 至 127 范围内的赋值,Integer 对象是在 IntegerCache.cache() 方法中产生的,会复用已有对象,这个区间内的 Integer 值可以直接使用 == 进行判断,但是这个区间之外的所有数据,都会在堆上产生,并不会复用已有对象,因此推荐使用 equals 方法进行值的判断。

让我们来看一下这样一段代码:

Integera=100, b=100, c=150, d=150;
System.out.println(a==b);
System.out.println(c==d);

有了上文的分析,我们很轻松地就能知道答案:true,false。这是因为对于 - 128 至 127 范围内的值,Integer对象会使用缓存,因此b会复用a对象,而超出了缓存区间的 150 则不会使用缓存,因此会创建新的对象。

从上文对自动装箱的分析我们可以得知,每次对Integer对象进行赋值都会调用其valueOf()方法,接下来我们从源码来分析Integer类的缓存机制:

/*** Returns an {@code Integer} instance representing the specified* {@code int} value.  If a new {@code Integer} instance is not* required, this method should generally be used in preference to* the constructor {@link #Integer(int)}, as this method is likely* to yield significantly better space and time performance by* caching frequently requested values.** This method will always cache values in the range -128 to 127,* inclusive, and may cache other values outside of this range.** @param  i an {@code int} value.* @return an {@code Integer} instance representing {@code i}.* @since  1.5*/publicstaticIntegervalueOf(inti) {
if (i>=IntegerCache.low&&i<=IntegerCache.high)
returnIntegerCache.cache[i+ (-IntegerCache.low)];
returnnewInteger(i);
}

可以得知,Integer的缓存机制是通过它内部的一个静态类IntegerCache来实现的,在IntegerCache.low和IntegerCache.high之间的值将直接使用IntegerCache的cache数组中缓存的值,否则会创建一个新的对象。同时,从源码的注释中我们还可以得知:如果不要求必须新建一个整型对象,缓存最常用的值(提前构造缓存范围内的整型对象),会更省空间,速度也更快。因此Integer的缓存是为了减少内存占用,提高程序运行的效率而设计的。

进一步了解IntegerCache的源码,我们还可以知道缓存的区间其实是可以设置的:

/*** Cache to support the object identity semantics of autoboxing for values between* -128 and 127 (inclusive) as required by JLS.** The cache is initialized on first usage.  The size of the cache* may be controlled by the {@code -XX:AutoBoxCacheMax=<size>} option.* During VM initialization, java.lang.Integer.IntegerCache.high property* may be set and saved in the private system properties in the* sun.misc.VM class.*/privatestaticclassIntegerCache {
// 最小值固定为-128staticfinalintlow=-128;
staticfinalinthigh;
staticfinalIntegercache[];
// 初始化缓存数组static {
// 最大值可以通过属性来配置inth=127;
// 获取系统配置里设置的值StringintegerCacheHighPropValue=sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue!=null) {
try {
inti=parseInt(integerCacheHighPropValue);
i=Math.max(i, 127);
// 最大的缓存数组大小为Integer.MAX_VALUEh=Math.min(i, Integer.MAX_VALUE- (-low) -1);
            } catch( NumberFormatExceptionnfe) {
// 如果属性值不能转换为int,就忽略它.            }
        }
high=h;
cache=newInteger[(high-low) +1];
intj=low;
// 为缓存数组赋值for(intk=0; k<cache.length; k++)
cache[k] =newInteger(j++);
// range [-128, 127] must be interned (JLS7 5.1.7)assertIntegerCache.high>=127;
    }
privateIntegerCache() {}
}

从注释中我们可以得知:

  • 缓存是为了支持自动装箱对象的标识语义,在自动装箱时可以复用这些缓存的对象,而 -128 到 127 的范围是基于JLS的要求而设定的
  • 首次使用时会初始化缓存。缓存的大小可以通过虚拟机参数 -XX:AutoBoxCacheMax=<size>} 来控制,在VM初始化期间,java.lang.Integer.IntegerCache.high可以设置并保存在sun.misc.VM类中的私有系统属性中(-Djava.lang.Integer.IntegerCache.high=<value>),未指定则为 127

更多细节可以参考 JLS 的Boxing Conversion 部分的相关描述。

3 Long

3.1 源码分析

有了上文对于Integer类的分析,Long内部的自动装箱、拆箱以及缓存机制也就大同小异了:

/*** Returns a {@code Long} instance representing the specified* {@code long} value.* If a new {@code Long} instance is not required, this method* should generally be used in preference to the constructor* {@link #Long(long)}, as this method is likely to yield* significantly better space and time performance by caching* frequently requested values.** Note that unlike the {@linkplain Integer#valueOf(int)* corresponding method} in the {@code Integer} class, this method* is <em>not</em> required to cache values within a particular* range.** @param  l a long value.* @return a {@code Long} instance representing {@code l}.* @since  1.5*/publicstaticLongvalueOf(longl) {
finalintoffset=128;
if (l>=-128&&l<=127) { // will cachereturnLongCache.cache[(int)l+offset];
    }
returnnewLong(l);
}

与Integer不同的是,Long的valueOf()方法并没有被要求缓存特定范围的值,而且通过LongCache我们可以得知Long的缓存范围被固定在了 -128 到 127 ,并不能通过参数来修改缓存的范围:

privatestaticclassLongCache {
privateLongCache(){}
// 缓存,范围从 -128 到 127,+1 是因为有个 0staticfinalLongcache[] =newLong[-(-128) +127+1];
static {
// 缓存 Long 值,注意这里是 i - 128 ,所以在拿的时候就需要 + 128for(inti=0; i<cache.length; i++)
cache[i] =newLong(i-128);
    }
}

3.2 面试题

为什么使用 Long 时,大家推荐多使用 valueOf 方法,少使用 parseLong 方法?

因为 Long 本身有缓存机制,缓存了 -128 到 127 范围内的 Long,valueOf 方法会从缓存中去拿值,如果命中缓存,会减少资源的开销,parseLong 方法就没有这个机制。

4 总结

通过本文,我们学习了Integer、Long的自动装箱、拆箱原理,还学习了它们内部的缓存机制,缓存是一种重要且常见的性能提升手段,在很多应用场景诸如Spring、数据库、Redis中都有广泛运用,因此了解它的机制对于我们的学习工作都有很大的帮助。本文分析了Integer、Long内部的一些巧妙设计,对于其他包装类来说也是互通的,本文就不再赘述了,有兴趣的读者可以自行去研究。

注:以上分析皆基于JDK 1.8版本来进行。

目录
相关文章
|
4天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
16 2
|
1月前
|
Java Apache Maven
Java百项管理之新闻管理系统 熟悉java语法——大学生作业 有源码!!!可运行!!!
文章提供了使用Apache POI库在Java中创建和读取Excel文件的详细代码示例,包括写入数据到Excel和从Excel读取数据的方法。
59 6
Java百项管理之新闻管理系统 熟悉java语法——大学生作业 有源码!!!可运行!!!
|
2月前
|
数据采集 运维 前端开发
【Java】全套云HIS源码包含EMR、LIS (医院信息化建设)
系统技术特点:采用前后端分离架构,前端由Angular、JavaScript开发;后端使用Java语言开发。
81 5
|
8天前
|
人工智能 监控 数据可视化
Java智慧工地信息管理平台源码 智慧工地信息化解决方案SaaS源码 支持二次开发
智慧工地系统是依托物联网、互联网、AI、可视化建立的大数据管理平台,是一种全新的管理模式,能够实现劳务管理、安全施工、绿色施工的智能化和互联网化。围绕施工现场管理的人、机、料、法、环五大维度,以及施工过程管理的进度、质量、安全三大体系为基础应用,实现全面高效的工程管理需求,满足工地多角色、多视角的有效监管,实现工程建设管理的降本增效,为监管平台提供数据支撑。
24 3
|
13天前
|
运维 自然语言处理 供应链
Java云HIS医院管理系统源码 病案管理、医保业务、门诊、住院、电子病历编辑器
通过门诊的申请,或者直接住院登记,通过”护士工作站“分配患者,完成后,进入医生患者列表,医生对应开具”长期医嘱“和”临时医嘱“,并在电子病历中,记录病情。病人出院时,停止长期医嘱,开具出院医嘱。进入出院审核,审核医嘱与住院通过后,病人结清缴费,完成出院。
45 3
|
19天前
|
JavaScript Java 项目管理
Java毕设学习 基于SpringBoot + Vue 的医院管理系统 持续给大家寻找Java毕设学习项目(附源码)
基于SpringBoot + Vue的医院管理系统,涵盖医院、患者、挂号、药物、检查、病床、排班管理和数据分析等功能。开发工具为IDEA和HBuilder X,环境需配置jdk8、Node.js14、MySQL8。文末提供源码下载链接。
|
22天前
|
移动开发 前端开发 JavaScript
java家政系统成品源码的关键特点和技术应用
家政系统成品源码是已开发完成的家政服务管理软件,支持用户注册、登录、管理个人资料,家政人员信息管理,服务项目分类,订单与预约管理,支付集成,评价与反馈,地图定位等功能。适用于各种规模的家政服务公司,采用uniapp、SpringBoot、MySQL等技术栈,确保高效管理和优质用户体验。
|
1月前
|
JSON 前端开发 Java
震惊!图文并茂——Java后端如何响应不同格式的数据给前端(带源码)
文章介绍了Java后端如何使用Spring Boot框架响应不同格式的数据给前端,包括返回静态页面、数据、HTML代码片段、JSON对象、设置状态码和响应的Header。
133 1
震惊!图文并茂——Java后端如何响应不同格式的数据给前端(带源码)
|
1月前
|
存储 前端开发 Java
Java后端如何进行文件上传和下载 —— 本地版(文末配绝对能用的源码,超详细,超好用,一看就懂,博主在线解答) 文件如何预览和下载?(超简单教程)
本文详细介绍了在Java后端进行文件上传和下载的实现方法,包括文件上传保存到本地的完整流程、文件下载的代码实现,以及如何处理文件预览、下载大小限制和运行失败的问题,并提供了完整的代码示例。
453 1
|
2月前
|
传感器 监控 数据可视化
【Java】智慧工地解决方案源码和所需关键技术
智慧工地解决方案是一种新的工程全生命周期管理理念。它通过使用各种传感器、数传终端等物联网手段获取工程施工过程信息,并上传到云平台,以保障数据安全。
81 7