jdk源码阅读笔记-Integer
public final class Integer extends Number implements Comparable<Integer>
Integer 由final修饰了,所以该类不能够被继承,同时 Integer 继承了Number类,因此可以将Integer转换成 int 、double、float、long、byte和short类型的数据,另外,也实现了comparable接口,因此Integer类也可以进行自然排序。
构造方法只有两个:
public Integer(int value) {
this.value = value;
}
public Integer(String s) throws NumberFormatException {
this.value = parseInt(s, 10);
}
我们主要看第二个构造方法,传入一个字符串,然后调用parseInt方法,接下来进入parseInt的源码:
public static int parseInt(String s, int radix)
throws NumberFormatException
{
/*
* WARNING: This method may be invoked early during VM initialization
* before IntegerCache is initialized. Care must be taken to not use
* the valueOf method.
*/ if (s == null) {
throw new NumberFormatException("null");
} if (radix < Character.MIN_RADIX) {
throw new NumberFormatException("radix " + radix +
" less than Character.MIN_RADIX");
} if (radix > Character.MAX_RADIX) {
throw new NumberFormatException("radix " + radix +
" greater than Character.MAX_RADIX");
} int result = ;
// //是否为负数
boolean negative = false;
int i = , len = s.length();
//这里加个负号是防止数据溢出,int的数值范围 -2的31次方到2的31次方减一
int limit = -Integer.MAX_VALUE;
//最小基数
int multmin;
//十进制数字
int digit; if (len > ) {
char firstChar = s.charAt();
//第一个字符小于0,有可能是"-","+"或其他字符
if (firstChar < '') { // Possible leading "+" or "-"
//为负数
if (firstChar == '-') {
negative = true;
limit = Integer.MIN_VALUE;
} else if (firstChar != '+')//非数字
throw NumberFormatException.forInputString(s); if (len == ) // Cannot have lone "+" or "-"
throw NumberFormatException.forInputString(s);
i++;
}
/**
* 最小基数,主要防止 result *= radix; 这个操作时数据过大
* 导致数据丢失的问题,因为所以带符号32位int类型整数为-2147483648~2147483647
*/
multmin = limit / radix;
while (i < len) {
// Accumulating negatively avoids surprises near MAX_VALUE
//转换十进制,这里获取的是radix进制下相对应的10进制数字,如:
//Character.digit('a',16),则返回 10;
//若输入字符不在进制的范围之内,则返回 -1:
//Character.digit('t',16),返回 -1,Character.digit('a',10),返回 -1
digit = Character.digit(s.charAt(i++),radix);
//返回-1说明字符非法
if (digit < ) {
throw NumberFormatException.forInputString(s);
}
//超过了数据范围
if (result < multmin) {
throw NumberFormatException.forInputString(s);
}
/**
*在转换时从高位向地位方向转换
*/
result *= radix;
if (result < limit + digit) {
throw NumberFormatException.forInputString(s);
}
result -= digit;
}
} else {
throw NumberFormatException.forInputString(s);
}
return negative ? result : -result;
}
这个方法中最核心的步骤是1、result *= radix; 2、result -= digit; 经过这两个步骤将字符串转换成数值类型。大概流程是这样的:
假如字符串"1234" 转换成int类型,result 的初始值为0,radix默认为10;
首先截取字符串的第一个字符1(这里忽略各种检查),经过第一步计算 result = 0*10 = 0;第二部计算 result = 0 - 1 = -1;
第一遍循环结束后,result 的值 变成了 -1
截取第二个字符 2 ,result = -1 * 10 = -10,result = -10 - 2 = -12;
截取第三个字符 3 ,result = -12 * 10 = -120,result = -120 - 3 = -123;
截取第四个字符 4 ,result = -123 * 10 = -1230 ,result = -1230-4 = -1234;
循环结束,此时result的值为 -1234,完成字符串向整数型的转换,返回是取反即可。
下面我将从一个面试题引出问题,然后通过阅读源码来解决这个问题。
public static void main(String[] args) {
Integer i1 = ;
Integer i2 = ;
Integer i3 = ;
Integer i4 = ;
Integer i5 = Integer.valueOf();
Integer i6 = Integer.valueOf();
Integer i7 = new Integer();
System.out.println("i1 == i2 的结果是:" + (i1 == i2));
System.out.println("i3 == i4 的结果是:" + (i3 == i4));
System.out.println("i5 == i6 的结果是:" + (i5 == i6));
System.out.println("i1 == i5 的结果是:" + (i1 == i5));
System.out.println("i1 == i7 的结果是:" + (i1 == i7));
}
运行结果为:
i1 == i2 的结果是:true
i3 == i4 的结果是:false
i5 == i6 的结果是:true
i1 == i5 的结果是:true
i1 == i7 的结果是:false
我们先来看第一和第二条结果,同样是比较两个相同数值,为什么会有不同的结果呢?接下我将通过源码来解释原因。
首先,我们通过编译获取到class文件的字节码

从图中我们可以看到,在执行 Integer i1 = 100 这条命令的时候,编译器会调用Integer中的静态方法 valueOf,接下来我们看看 valueOf方法是怎么实现的吧。
public static Integer valueOf(int i) {
if (i >= IntegerCache.low && i <= IntegerCache.high)
return IntegerCache.cache[i + (-IntegerCache.low)];
return new Integer(i);
}
这个代码看起来很简单,Integer 中有一个静态内部类 IntegerCache,调用该方法时首先会判断该值是否在缓存的范围内,如果在则直接将缓存中的数值返回,否则返回一个新对象。看到这里我们似乎已经知道了上面的问题的答案了,接下来继续看静态内部类吧
private static class IntegerCache {
static final int low = -;
static final int high;
static final Integer cache[];
static {
// high value may be configured by property
//缓存范围最小(也是默认范围)为 (-128)~ 127,如果配置java.lang.Integer.IntegerCache.high
//high 的值可从配置文件中读取
int h = ;
String integerCacheHighPropValue =
sun.misc.VM.getSavedProperty("java.lang.Integer.IntegerCache.high");
if (integerCacheHighPropValue != null) {
try {
int i = parseInt(integerCacheHighPropValue);
//获取配置文件和127之间的最大值
i = Math.max(i, );
// Maximum array size is Integer.MAX_VALUE
//最大值范围
h = Math.min(i, Integer.MAX_VALUE - (-low) -);
} catch( NumberFormatException nfe) {
// If the property cannot be parsed into an int, ignore it.
}
}
high = h;
//创建缓存数组
cache = new Integer[(high - low) + ];
int j = low;
//将数字缓存起来默认 -128 ~ 127
for(int k = ; k < cache.length; k++)
cache[k] = new Integer(j++);
// range [-128, 127] must be interned (JLS7 5.1.7)
assert IntegerCache.high >= ;
}
private IntegerCache() {}
}
我们知道内部类只有在所在类实例化时才会被实例化,而且只会实例化一次,缓存操作是在静态代码块中完成,也就是说在类被实例化的时候数据就已经被缓存好了,接下使用的时候可以直接使用缓存的数据。
现在我们回归到上面的问题,结果1中两个数据均为 100,在缓存的范围中,因此i1和i2都指向的是同一个内存地址,因此返回true。结果2中 两个数都是200,超出了缓存的范围,所以直接new 出了两个对象,因此他们的内存地址不一致,返回结果为false;另外,使用valueOf 和 使用 = 操作符赋值时一样的,所以结果3和结果4返回结果为 true,结果5中 是直接使用new关键字创建对象,所以他们的内存地址肯定不一致,结果为false。
那么,现在问题又来了,那我怎么判断两个整数的大小呢?继续看源码
/**
* The value of the {@code Integer}.
*
* @serial
*/
private final int value;
public boolean equals(Object obj) {
if (obj instanceof Integer) {
return value == ((Integer)obj).intValue();
}
return false;
}
public int intValue() {
return value;
}
是的,没错,比较两个数值大小时可以使用equals方法来比较,源码中value的类型为 int型,intValue返回的也是value,因此可以判断两个数的大小。
public static void main(String[] args) {
Integer i1 = ;
Integer i2 = ;
System.out.println("i1 == i2 的结果是:" + i1.equals(i2)); //true
}
补充:equals 与 == 的区别:
equals 比较的是两个数值的大小,== 有两种情况,如果比较的是 基本数据类型,则 == 跟equals一样都是比较的大小,如果是引用类型或数组,则比较是内存地址。
getChars方法:
static void getChars(int i, int index, char[] buf) {
int q, r;
int charPos = index;
char sign = ;
if (i < ) {
sign = '-';
i = -i;
}
// Generate two digits per iteration
//每次循环获取后两位数
while (i >= ) {
q = i / ;
// really: r = i - (q * 100);
//使用位移运算的效率高于乘法运算,r为后两位数
r = i - ((q << ) + (q << ) + (q << ));
i = q;
//获取后两位数的个位
buf [--charPos] = DigitOnes[r];
//十位
buf [--charPos] = DigitTens[r];
}
// Fall thru to fast mode for smaller numbers
// assert(i <= 65536, i);
//每次只取个位数
for (;;) {
//相当于i*(52429/524288)=i*0.10000038146972656=i*0.1=i/10
//这里选 52429 和 2的19次方相除,得到的结果精度更加高,更加接近于 i/10的结果
//之所以要这样转换,是因为在计算机运算中位移的效率 > 乘法效率 > 除法效率
q = (i * ) >>> (+);
r = i - ((q << ) + (q << )); // r = i-(q*10) ...
buf [--charPos] = digits [r];
i = q;
if (i == ) break;
}
if (sign != ) {
buf [--charPos] = sign;
}
}
jdk源码阅读笔记-Integer的更多相关文章
- jdk源码阅读笔记-LinkedHashMap
Map是Java collection framework 中重要的组成部分,特别是HashMap是在我们在日常的开发的过程中使用的最多的一个集合.但是遗憾的是,存放在HashMap中元素都是无序的, ...
- JDK源码学习笔记——Integer
一.类定义 public final class Integer extends Number implements Comparable<Integer> 二.属性 private fi ...
- jdk源码阅读笔记-ArrayList
一.ArrayList概述 首先我们来说一下ArrayList是什么?它解决了什么问题?ArrayList其实是一个数组,但是有区别于一般的数组,它是一个可以动态改变大小的动态数组.ArrayList ...
- jdk源码阅读笔记-HashSet
通过阅读源码发现,HashSet底层的实现源码其实就是调用HashMap的方法实现的,所以如果你阅读过HashMap或对HashMap比较熟悉的话,那么阅读HashSet就很轻松,也很容易理解了.我之 ...
- jdk源码阅读笔记
1.环境搭建 http://www.komorebishao.com/2020/idea-java-jdk-funyard/ 2. 好的源码阅读资源 https://zhuanlan.zhihu.co ...
- jdk源码阅读笔记-HashMap
文章出处:[noblogs-it技术博客网站]的博客:jdk1.8源码分析 在Java语言中使用的最多的数据结构大概右两种,第一种是数组,比如Array,ArrayList,第二种链表,比如Array ...
- jdk源码阅读笔记-LinkedList
一.LinkedList概述 LinkedList的底层数据结构为双向链表结构,与ArrayList相同的是LinkedList也可以存储相同或null的元素.相对于ArrayList来说,Linke ...
- jdk源码阅读笔记-AbstractStringBuilder
AbstractStringBuilder 在java.lang 包中,是一个抽象类,实现 Appendable 接口和 CharSequence 接口,这个类的诞生是为了解决 String 类在创建 ...
- jdk源码阅读笔记-String
本人自学java两年,有幸初入这个行业,所以功力尚浅,本着学习与交流的态度写一些学习随笔,什么错误的地方,热烈地希望园友们提出来,我们共同进步!这是我入园写的第一篇文章,写得可能会很乱. 一.什么是S ...
随机推荐
- linux小实验-考勤模拟程序
任务: 设计一个考勤模拟程序,实现如下功能选择界面,要求使用函数 1.上班签到 2.下班签出 3.缺勤信息查阅 4.退出 考勤程序运行后,提示用户输入上述功能选择,并验证用户输入的用户名和密码:用户信 ...
- ES入门笔一
ES6一共有6种声明变量的方法 --ES5只有var 和 function --ES6新增了let.const.import和class四种 ES6新增let和const,用来声明变量,是对var的扩 ...
- Mysql 查询条件中字符串尾部有空格也能匹配上的问题
一.表结构 TABLE person id name 1 你 2 你(一个空格) 3 你(二个空格) 二.查询与结果 select * from person where `name` = ? 无论 ...
- 分布式爬虫框架XXL-CRAWLER
<分布式爬虫框架XXL-CRAWLER> 一.简介 1.1 概述 XXL-CRAWLER 是一个分布式爬虫框架.一行代码开发一个分布式爬虫,拥有"多线程.异步.IP动态代理.分布 ...
- Html5的表单元素
表单是HTML中获取用户输入的手段,,对于web应用系统及其重要,文字是不能说明问题的: 直接上代码把: <!DOCTYPE html><html lang="en&quo ...
- tomcat的配置使用详细版
摘要: 开发者开发部署web应用时通常使用tomcat服务器,很多初学者只懂得在开发工具上配置,但离开了开发工具,自己手动配置部署,并让一个项目跑起来,你会了吗.小编也遇到过这样的困扰.网上查找的资料 ...
- 树莓派配置watchdog
安装watchdog apt install watchdog 编辑/etc/modules,添加bcm2708_wdog 编辑/etc/watchdog.conf watchdog-device = ...
- C# 发展史
C# 发展史 Intro 本文主要总结介绍C# 每个版本带来的不同的语言特性. C#,读作C Sharp,是微软推出的一种基于.NET平台的.面向对象的高级编程语言.是微软公司在2000年发布的一种新 ...
- Java CAS 原理分析
1.简介 CAS 全称是 compare and swap,是一种用于在多线程环境下实现同步功能的机制(可以把 CAS 看做乐观锁).CAS 操作包含三个操作数 -- 内存位置.预期数值和新值.CAS ...
- 5月11日——IOS下如何检测用户是否安装微信
执行如下代码: var WXApi = plus.ios.import("WXApi"); var isWXInstalled = WXApi.isWXAppInsta ...