欢迎关注方志朋的博客,回复”666“获面试宝典
文章来源:https://c1n.cn/MSqAy
目录
- 背景
- 事故
- 分析
- 总结
- 工具分享
我们在使用金额计算或者展示金额的时候经常会使用 BigDecimal,也是涉及金额时非常推荐的一个类型。
BigDecimal 自身也提供了很多构造器方法,这些构造器方法使用不当可能会造成不必要的麻烦甚至是金额损失,从而引起事故资损。
事故
接下来我们看下收银台出的一起事故。
| 问题描述 收银台计算商品金额报错,导致订单无法支付。 | 事故级别 P0 | 事故过程 如下:
- 13:44,接到报警,订单支付失败,支付可用率降至 60%
- 13:50,迅速回滚上线代码,恢复正常
- 14:20,review 代码,预发布验证发现问题点
- 14:58,修改问题代码上线,线上恢复
| 故障原因 BigDecimal 在金额计算中丢失精度。 原因分析
首先我们先用一段代码复现问题根源,如下所示:
public static void main(String[] args) {
BigDecimal bigDecimal=new BigDecimal(88);
System.out.println(bigDecimal);
bigDecimal=new BigDecimal("8.8");
System.out.println(bigDecimal);
bigDecimal=new BigDecimal(8.8);
System.out.println(bigDecimal);
}
执行结果如下:
文章图片
通过测试发现,当使用 double 或者 float 这些浮点数据类型时,会丢失精度,String、int 则不会,这是为什么呢?
我们点开构造器方法看下源码:
public static long doubleToLongBits(double value) {
long result = doubleToRawLongBits(value);
// Check for NaN based on values of bit fields, maximum
// exponent and nonzero significand.
if ( ((result & DoubleConsts.EXP_BIT_MASK) ==
DoubleConsts.EXP_BIT_MASK) &&
(result & DoubleConsts.SIGNIF_BIT_MASK) != 0L)
result = 0x7ff8000000000000L;
return result;
}
问题就处在 doubleToRawLongBits 这个方法上,在 jdk 中 double 类(float 与 int 对应)中提供了 double 与 long 转换,doubleToRawLongBits 就是将 double 转换为 long,这个方法是原始方法(底层不是 java 实现,是 c++ 实现的)。
double 之所以会出问题,是因为小数点转二进制丢失精度。
文章图片
BigDecimal 在处理的时候把十进制小数扩大 N 倍让它在整数上进行计算,并保留相应的精度信息。
①float 和 double 类型,主要是为了科学计算和工程计算而设计的,之所以执行二进制浮点运算,是为了在广泛的数值范围上提供较为精确的快速近和计算。
②并没有提供完全精确的结果,所以不应该被用于精确的结果的场合。
③当浮点数达到一定大的数,就会自动使用科学计数法,这样的表示只是近似真实数而不等于真实数。
④当十进制小数位转换二进制的时候也会出现无限循环或者超过浮点数尾数的长度。
总结
所以,在涉及到精度计算的过程中,我们尽量使用 String 类型来进行转换。
热门内容:
- 比 MyBatis 效率快 100 倍的条件检索引擎,天生支持联表!
- MySQL 暴跌!
- 到处都开始显示地区了,这是好事儿啊!
- 减少 try catch ,可以这样干!
- 手下两个应届生:一个踏实喜欢加班,一个技术强挑活,怎么选??
- IDEA公司再发新神器!超越 VS Code 骚操作!
- 放弃密码模式吧,最先进的Spring Cloud认证授权方案在这里
文章图片
最近面试BAT,整理一份面试资料《Java面试BAT通关手册》,覆盖了Java核心技术、JVM、Java并发、SSM、微服务、数据库、数据结构等等。
获取方式:点“在看”,关注公众号并回复 666 领取,更多内容陆续奉上。
【数据结构|BigDecimal使用不当,造成P0事故!】明天见(??ω??)??
推荐阅读
- 数据库|MySQL 暴跌!
- 极致效率,云原生数据库TDSQL-C安身立命的根本
- 数据湖构建—如何构建湖上统一的数据权限
- 现代斗山X瓴羊(“一横四纵“解决方案聚焦中台场景级部署)
- Java学习|Spring boot微服务架构中,利用RestTemplate调用别人提供的接口
- 数据库|MySQL数据库之索引,事务,与存储引擎
- mybatis|一种投机取巧方式基于Mybatis-plus实现Mysql动态表(待完善)
- 数据库|MySQL数据库之管理与基本SQL语句
- java|JavaWeb学习心得