Leader:这样的 Bug 你也写的出来???
Hello~各位读者新年好!不知道大家春节假期是否已延长,小黑哥刚接到通知,假期延长到 2 月 2 号,另外回去之后需要在家办公,自行隔离两周。还没试过在家办公,小黑哥就怕到时候生物钟还没调整过来,一觉睡醒已经是下午了。。。
前言
春节假期,还躺在床上小黑哥,收到对账系统的一条预警短信,提示当前系统资金核对存在问题。关于资金的问题,都是大问题,小黑哥连忙拔出电脑,连上 VPN,登录生产环境的查看相关日志。
通过日志,很快小黑哥定位到相关代码。
有的同学可能一下子就能看出这里的问题, Long
对象采用 !=
进行比较,这真是一个低级 Bug
。幸好 Leader 还不知道,赶紧悄悄修复一下。
现在回想小黑哥当初写这段代码的时候,误以为两个 Long
对象比较将会进行自动拆箱,转变为两个基本数值类型比较。
下面开始复习一下 Java
自动装箱与拆箱机制。
自动装箱与拆箱机制
自动装箱(Autoboxing),是 JDK5
新增的一种语法糖,将会在代码编译时自动将原始类型转换为其对应的对象包装器类。例如将 int
转换为 Integer
,double
转换为 Double
。如果转换结果相反,我们就将其称为拆箱。
下面是一个自动装箱的例子:
上面代码 li.add(i)
就发生自动装箱,将基本数据类型 long
转换为其包装类 Long
。
查看这段代码对应的字节码。
图上黄线标注的字节码对应的代码为 li.add(i)
。从这我们可以看到 long
类型的自动装箱实际上调用 Long#valueOf
方法。所以编译器运行时将之前的代码转换为下面的代码
接下来我们来看一个自动拆箱的例子:
由于 Long
包装类对象不能用于求模(%
)以及 +=
,所以这段代码将会发生自动拆箱,将 Long
转化为 long
类型。相应的这里我们也看下其编译之后的字节码。
这里的字节码比之前的复杂很多,这里主要关注黄线部分。可以看到这里调用 Long#longValue
将 Long
对象转为 long
类型。所以自动拆箱这个例子,最后编译器生成字节码等同于以下代码:
Java
规定的 8 种数据类型都有其对应包装类,这些都可以进行相应的自动装箱和拆箱。
这里小结一下:
- 自动装箱机制是通过调用包装器类
valueOf
实现 - 自动拆箱机制通过调用包装器类的相应的
**Value
,如longValue
,intValue
实现
Cache 陷阱
自动装箱和拆箱概念说起其实挺简单的,但是如果使用不当可能就会踩坑。
我们来看一段代码:
如果你对上面的结果的不是很清楚,恭喜你,暖男小黑哥帮你排雷了。
上面输出结果为:
true
false
这里输出结果之所以为这样,主要与 LongCache
有关。自动装箱进制将会调用 Long#valueOf
,其源码如下:
只要数值范围位于 [-128,127] 之间,valueOf
就会返回 LongCache.cache
这个数组中的值。所以 aLong/bLong
其实是同一个对象,cLong/dLong
是两个不用对象的。
Long#valueOf
这个方法通过 Cache
减少创建对象的数量,提高相应的空间以及时间性能。
至于为什么缓存 [-128,127] 之间的数字,而没有缓存更多的值,甚至缓存所有的值?
这是因为 Long
范围为[-263,263],总共 2^64 ,这么多对象实例,显然不宜全部缓存。至于 [-128,127] 之间数字 JDK
设计者认为这些数字使用频率高,个人认为这是一个经验值。
另外这个 cache
可能会导致另外一个问题:锁共用。
上面代码看起来 A
类使用 along
这个对象锁,而 B
类使用 blong
这个对象锁,看起来两个风马牛不相及,但是实际上两个对象由于 Cache
机制,导致其实际上使用同一个对象,A
与 B
共用同一把锁。
除了 Long#valueOf
方法中存在 Cache
以外,Byte
,Short
,Integer
,Character
也存在相应的 Cache
值。其中 Integer
对象可以通过 -XX:AutoBoxCacheMax=<size>
改变 Cache
初始范围。
技术总结
通过自动装箱与拆箱机制,大大减少了数据类型转化的代码,但是同时带来一些隐藏的问题。这里我们需要记住:
- 所有对象之间不能使用
==
比较,需要使用equals
代替,推荐使用JDK7
提供的Objects#equals
方法,该方法可以有效避免比较过程空指针问题 - 基本数据类型的包装类不适合当做锁对象
各位读者朋友们,请牢记这两个总结,不要踩中这两个坑哦。另外推荐大家安装一下 FindBugs
这个插件,通过这个插件扫描代码,可以找出代码中的相关缺陷。上面两个问题,通过 FindBugs
都能扫描出来,简直神器。
对象比较问题:
锁问题
帮助文档
- Autoboxing and Unboxing『https://docs.oracle.com/javase/tutorial/java/data/autoboxing.html』
- 极客时间-『Java并发编程实战』专栏
Leader:这样的 Bug 你也写的出来???的更多相关文章
- 今天写代码遇到了一个BUG
在我们日常写注释是需要注意,在有返回值的里面不要用以下方式注释,不然不会报错,运行状态码返回也是200,但就不是想要的结果. 下面举个例子 return { 'token':token, 'user_ ...
- 修改BUG心得
修改BUG心得 分类: 项目管理/CMMI2013-01-14 22:06 845人阅读 评论(0) 收藏 举报 目录(?)[-] 一 二 三 一. 1.写第一版时就杜绝这些的发生. 2.思维要开 ...
- [中英对照]The Art Of Reporting Bugs | 报bug的艺术
前言:因为最近要给兄弟Team分享一下如何有效地报告bug, 故多做一做功课.下面给出一篇博客的中英文对照翻译. The Art Of Reporting Bugs | 报bug的艺术 My init ...
- 做一名开源社区的扫地僧——从Bug report到Google Summer of Code(GSoC):从200个bug到5000美金
今年的软件自由日(SFD),我在广州Linux用户组的线下活动上做了一个分享,主题叫做<做一名开源社区的扫地僧(上)>.我把演讲的内容重新整理扩充, 写出了文字版, 希望可以跟更多朋友分享 ...
- margin和padding那点事及常见浏览器margin padding相关Bug
用Margin还是用Padding 何时应当使用margin: 需要在border外侧添加空白时. 空白处不需要背景(色)时. 上下相连的两个盒子之间的空白,需要相互抵消时.如15px + 20px的 ...
- 正则转nfa:bug出现。
本人写的一个正则到nfa的bug 刚写完前面的那篇,自己用脑子过了一下,发现了一个bug.具体情况如下. 这个bug的产生条件是多次调用假名的时候,每次调用都会修改假名的nfa图.直接这么说不好理解, ...
- IE6中的常见BUG与相应的解决办法
开发前端的同学一定都知道,IE6是兼容BUG最多的浏览器,它不支持PNG alpha通道暂且不论.其文档的解析理解规范也引起了诸多恼人的BUG,有时甚至让人感到绝望.本文主要讲解一些比较容易遇到的IE ...
- 记一次亲身踩过的hibernate的bug
记一次亲身踩过的hibernate的bug 在写实体类时,经常会对域增加校验,例如@NotNull表示哪个字段不能为空,昨天晚上调试代码,就遇到了问题, @Entity public class Ap ...
- 客户端一致性与多Leader机制------《Designing Data-Intensive Applications》读书笔记7
接着上一篇的内容,我们继续来梳理分布式系统之中的副本机制与副本一致.上文我们聊到了在可用性与一致性之间的一个折中的一致性等级:最终一致性.我们顺着上篇的内容,由用户来分析一致性等级. 1. 客户端的困 ...
随机推荐
- 关于Win32串口
因为近段时间接触Hid相对来说多一些,由此忽略了串口中获取cbInQue这个重要的东西,下面是错误代码 // Win32SerialPortLib.cpp : 定义 DLL 应用程序的导出函数. // ...
- Jquery拖拽,拖动排序插件
上github搜jquery-sortable <!-- jq拖拽排序 --> <script src="${contextPath}/static/excelTable/ ...
- bzoj3744: Gty的妹子序列 (BIT && 分块)
强制在线的区间询问逆序对数 如果不是强制在线 就是可以用莫队乱搞啦 强制在线的话 用f[i][j]记录第i块到第j个点之间的逆序对数 用s[i][j]记录前i块中小于等于j的数字个数 离散化一下 BI ...
- java程序员摸爬滚打的三年,这些经历你值得借鉴
不知不觉都2020年2月底了,小羊同学从毕业快开始都一直从事java开发这个行业,前两天开通了头条号,想借此发文分享一下这几年的经历吧,如果你是还没毕业的大学生或者刚入行不久,也许会对你有帮助. 1: ...
- update_jz首项V5.0-Tutorial
What's New: 增加了4个对话框,用于展示信息.归并条目.剔除条目 增加了可视化统计图形中每个科室(柱形)的统计总数 可视化图形一些颜色调整(无奈在省份很多的条件下一些颜色还不易区分) 下面是 ...
- night of 2019.8.14
饭后一个小时,坐下来看了一会书,是原来的同事推荐的. 最近一段时间,心一点一点平静.能平下心来去做一些事.偶尔也常常对着窗户发呆,想一些人生感悟.想到一些感悟时,常常感到没有人可以说说心里话. 丰子恺 ...
- Iris路由和路由组
package main import ( "github.com/kataras/iris" "github.com/kataras/iris/context" ...
- mediasoup-demo安装记录
环境CentOS 7 64位 VMWare12虚拟机(win10主机),安装好NodeJS 10.13(大于8.9就可以) 已按照GitHub说明拉下来代码,配置好Node环境,开始执行npm sta ...
- codeforces 1245D(最小生成树)
题面链接:https://codeforces.com/problemset/problem/1245/D 题意大概是给你一些城市的坐标,可以在城市中建立发电站,也可以让某个城市和已经建好发电站的城市 ...
- Bugku-CTF之求getshell
Day31 求getshell http://123.206.87.240:8002/web9/