两个Integer对象比较大小,为什么100等于100,1000不等于1000 ?

简介: 前几天,有位小伙伴向我反馈,在维护代码过程中,出现了一个莫名其妙的问题。明明上线之后程序跑得还好好的,可程序上线运行一段时间之后,所有,代码没有做任何修改,发 cxccccc现运行结果和期望值恰好相反。因为涉及到金额造成了比较大的损失,最后,这位小伙伴还被公司辞退了,大家可以来评论一下,这位小伙伴背的这个锅值不值?

需要面试文档的扫描下方二维码

前几天,有位小伙伴向我反馈,在维护代码过程中,出现了一个莫名其妙的问题。明明上线之后程序跑得还好好的,可程序上线运行一段时间之后,所有,代码没有做任何修改,发 cxccccc现运行结果和期望值恰好相反。因为涉及到金额造成了比较大的损失,最后,这位小伙伴还被公司辞退了,大家可以来评论一下,这位小伙伴背的这个锅值不值?

1、业务场景

大家来看,他的代码大致是这样写的:

0066855b8250418b6c68fa4c8fb76767.jpg

一般情况下,a和b都输入100的时候,返回为true,但当a和b都输入1000的时候,返回为false。按照正常逻辑理解,100 等于等于 100,那1000 为什么不等于等于1000 呢?这位同学,百思不得其解。于是,这位同学,还特意写了一段测试代码

124792a1900dadfd005da3a4fb39a365.jpg

这到底是什么原因呢?我们对照Integer的源码来进行分析:

2、源码分析

我摘取了Integer的源码片段,它有一个valueOf()的方法:

831d8732d85f72842d0d1375f09aa44d.jpg

我们可以看到,Integer源码中的valueOf()方法做了一个条件判断,其中 IntegerCache.low的值为-128,

bc9d1c0082f9a6ee4ce8ed66d3680815.jpg

IntegerCache.high的值为127。

97df389f7e4bc5b3e857c9e4d97f13c6.jpg

也就是说如果目标值在-128~127之间,会直接从cache数组中取值,否则就会新建对象。

这里又有人会问了,那为什么默认是-128 - 127,怎么不是-200 - 200或者是其他值呢?那JDK为何要这样做呢?

d1a33d4116b549c3c0ef3fce2d0f7afe.jpg

在Java API 中是这样解释的:

Returns an Integer instance representing the specified int value. If a new Integer instance is not required, this method should generally be used in preference to the constructor 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

大致意思是:

-128~127的数据在int范围内是使用最频繁的,为了减少频繁创建对象带来的内存消耗,这里其实是用到了享元模式,以提高空间和时间性能。

在JDK中,这样的应用不止int,我给小伙伴们整理了一个表格,表格中的其他类型也都应用了享元模式,也就是说对数值做了缓存,只是缓存的范围不一样,具体如表中所示:

88e05342da5b54deed1acad09bf14c11.jpg

我需要这张表的小伙伴可以私信我,以上就是关于Integer对象比较大小的分析,听懂了的小伙伴,请关注点个赞,下次不迷路。

我是被编程耽误的文艺Tom,如果大家还有其他疑问,请在评论区留言。如果本次面试解析对你有帮助,请动动手指一键三连分享给更多的人。关注我,面试不再难!

14.为什么HashMap会产生死循环?

HashMap死循环是一个比较常见、也是比较经典的面试题,在大厂的面试中也经常被问到。HashMap的死循环问题只在JDK1.7版本中会出现,主要是HashMap自身的工作机制,再加上并发操作,从而导致出现死循环。JDK1.8以后,官方彻底解决了这个问题。

1、数据插入原理

在分析原因之前,我先带大家了解一下JDK1.7中HashMap插入数据的原理,来看动画演示:

f1a87e69a9662a4611e656bce251475c.jpg

由于JDK 1.7中HashMap的底层存储结构采用的是数组 加 链表的方式。

7af4d022debbac1840c52436fdfe1a24.jpg

而HashMap在数据插入时又采用的是头插法,也就是说新插入的数据会从链表的头节点进行插入。

13a1d6e7c60a27bfbb1c612f2aa92f56.jpg

因此,HashMap正常情况下的扩容就是是这样一个过程。我们来看,旧HashMap的节点会依次转移到新的HashMap中,旧HashMap转移链表元素的顺序是A、B、C,而新HashMap使用的是头插法插入,所以,扩容完成后最终在新HashMap中链表元素的顺序是C、B、A。

2、导致死循环的原因

接下来,我通过动画演示的方式,带大家彻底理解造成HashMap死循环的原因。我们按以下三个步骤来还原并发场景下HashMap扩容导致的死循环问题:

85af3b2cd8036177851e54a5da58da95.jpg

第一步:线程启动,有线程T1和线程T2都准备对HashMap进行扩容操作, 此时T1和T2指向的都是链表的头节点A,而T1和T2的下一个节点分别是T1.next和T2.next,它们都指向B节点。

ffcafb9fd212b3be89abaad7cbfe2093.jpg

第二步:开始扩容,这时候,假设线程T2的时间片用完,进入了休眠状态,而线程T1开始执行扩容操作,一直到线程T1扩容完成后,线程T2才被唤醒。

d668355d0cca7b9bb5d68b53ca33da79.jpg

T1完成扩容之后的场景就变成动画所示的这样。

876a16bb38261377c4a16fde751cfbe5.jpg

因为HashMap扩容采用的是头插法,线程T1执行之后,链表中的节点顺序发生了改变。但线程T2对于发生的一切还是不可知的,所以它指向的节点引用依然没变。如图所示,T2指向的是A节点,T2.next指向的是B节点。

ed02e2dd9dda0b9a49b5ef330c3519b3.jpg

当线程T1执行完成之后,线程T2恢复执行时,死循环就发生了。

d0e31ab26ea867a31a6283a239ce62df.jpg

因为T1执行完扩容之后,B节点的下一个节点是A,而T2线程指向的首节点是A,第二个节点是B,这个顺序刚好和T1扩容之前的节点顺序是相反的。T1执行完之后的顺序是B到A,而T2的顺序是A到B,这样A节点和B节点就形成了死循环。

3、解决方案

避免HashMap发生死循环的常用解决方案有三个:

1)、使用线程安全的ConcurrentHashMap替代HashMap,个人推荐使用此方案。

2)、使用线程安全的容器Hashtable替代,但它性能较低,不建议使用。

3)、使用synchronized或Lock加锁之后,再进行操作,相当于多线程排队执行,也会影响性能,不建议使用。

4、总结

HashMap死循环只发生在JDK1.7版本中,主要原因是JDK1.7中的HashMap,在头插法 加 链表 加 多线程并发 加 扩容这几个情形累加到一起就会形成死循环。多线程环境下建议采用ConcurrentHashMap替代。在JDK1.8中,HashMap改成了尾插法,解决了链表死循环的问题。

以上就是关于HashMap死循环原因的分析,听懂的小伙伴,关注点个赞,下次不迷路。

本文为“Tom弹架构”原创,转载请注明出处。技术在于分享,我分享我快乐!

如果本文对您有帮助,欢迎关注和点赞;如果您有任何建议也可留言评论或私信,您的支持是我坚持创作的动力。

相关文章
让用户输入x的值,如果x的值如果小于1,y=x,x如果大于等于1并且小于10,y=2x。。。 // x x < 1 // y={ 2X 1<=x
让用户输入x的值,如果x的值如果小于1,y=x,x如果大于等于1并且小于10,y=2x。。。 // x x < 1 // y={ 2X 1<=x
102 0
输入一个整数,判断大于0小于0还是等于0
输入一个整数,判断大于0小于0还是等于0
|
6月前
|
SQL
条件构造器,MybatisPlus支持各种复杂的where条件,其实就是Wrapper,eq是等于的意思,相当于等于那个数值,ne就是不等于,gt大于的意思,ge大于等于,QueryWrapper是做
条件构造器,MybatisPlus支持各种复杂的where条件,其实就是Wrapper,eq是等于的意思,相当于等于那个数值,ne就是不等于,gt大于的意思,ge大于等于,QueryWrapper是做
|
2月前
判断最大值
【10月更文挑战第31天】判断最大值。
43 7
|
8月前
|
存储 前端开发 JavaScript
为什么0.1 + 0.2 不等于 0.3 ?
这篇编程技术文章探讨了为什么在多种编程语言中,0.1 + 0.2 不等于 0.3 的现象。问题源于计算机使用二进制浮点数表示小数,而二进制无法精确表示某些十进制分数,如 0.1 和 0.2。这导致它们在计算机内部被近似表示,从而在相加时产生微小误差。文章通过示例和图片解释了二进制浮点数的表示原理,并提供了将小数转换为整数再相加以及使用 `toFixed()` 方法或 decimal.js 库等解决精度问题的方法。
101 2
|
8月前
|
算法 测试技术 C#
【状态机dp】【 排序 】 2809使数组和小于等于 x 的最少时间
【状态机dp】【 排序 】 2809使数组和小于等于 x 的最少时间
|
8月前
位与运算&小于相等判断==的优先级
位与运算&小于相等判断==的优先级
|
C++
37 C++ - 等于和不等于(==、!=)运算符重载
37 C++ - 等于和不等于(==、!=)运算符重载
61 0
变量等于0时的判断问题
变量等于0时的判断问题
88 0
有一个整数数组,长度为9,数组里的值是多少不清楚,但是知道数组中有8个值是相等,其中一个小于其他8个值,目前有一个标准函数,compare(int[] a, int[] b),返回0相等1大于
有一个整数数组,长度为9,数组里的值是多少不清楚,但是知道数组中有8个值是相等,其中一个小于其他8个值,目前有一个标准函数,compare(int[] a, int[] b),返回0相等1大于
130 5