Java BigDecimal解决double精度丢失

我们在日常开发中, 有很多时候会遇到小数(double类型)精确计算, 由于 double 类型在计算机中的特殊存储方式, 某些情况下会导致精度丢失, 特别是跟钱有关的计算, 后台校验的时候, 那得是分毫不差.

场景

  • 商品a价格是 0.1, 商品b 价格是 0.2, 用户买了这两个商品, 付款时, 订单总金额是 0.3, 如果用 double 类型做计算, 将会得到结果 0.30000000000000004, 前端如果拿来展示总金额, 用户肯定就蒙圈了, 这显然不是我们想要的.
  • 那么如何解决呢? Java 给我们提供了专门解决这个问题的类 BigDecimal.
ini 复制代码
double a = 0.1;
double b = 0.2;
double c = a + b;
System.out.println(c); // 0.30000000000000004

BigDecimal 使用

创建 BigDecimal 对象

创建对象一共有3种方式, 两个构造器, 一个静态方法, 实际静态方法是对入参 String 的构造器的封装, 即 2, 3 两种方式本质上是一样的.

ini 复制代码
 // 1 入参是 double
 BigDecimal a = new BigDecimal(0.3);
 // 2 入参是 String
 BigDecimal b = new BigDecimal("0.3");
 // 3 静态方法, 入参是 double
 BigDecimal c =  BigDecimal.valueOf(0.3);

计算操作

  • 公用代码
ini 复制代码
double a = 0.1;
double b = 0.2;
BigDecimal a1 = BigDecimal.valueOf(a);
BigDecimal b1 = new BigDecimal(Double.toString(b));
1. 加法 add
  • 这就很好的解决了上面场景中出现的问题
ini 复制代码
BigDecimal c1 = a1.add(b1);  
System.out.println(c1); // 0.3
2. 减法 subtract
ini 复制代码
BigDecimal c2 = a1.subtract(b1);  // 减法
System.out.println(c2); // -0.1
3. 乘法
ini 复制代码
BigDecimal c3 = a1.multiply(b1);  // 乘法
System.out.println(c3); // 0.02
4. 除法
ini 复制代码
BigDecimal c4 = a1.divide(b1);  // 除法  0.1 / 0.3, 无法除尽, 报错.
System.out.println(c4); // 0.5

BigDecimal c5 = a1.divide(c1);  // 除法  
System.out.println(c5); // 0.1 / 0.3, 无法除尽, 报错.

c5 = a1.divide(c1, 2, RoundingMode.HALF_UP);  // 除法  0.1 / 0.3  四舍五入
System.out.println(c5); // 0.5

注意:

  • BigDecimal 进行除法计算时, 强烈建议使用指定保留小数倍数和舍入模式这个方法, 这样可以避免除不尽造成的异常.
arduino 复制代码
/**
* 参数1: 除数
* 参数2: 保留小数的位数
* 参数3: 舍入模式
*/
public BigDecimal divide(BigDecimal divisor, int scale, RoundingMode roundingMode)
5. BigDecimal 转化为 double
ini 复制代码
double value = c4.doubleValue();

总结

使用 BigDecimal 计算时要注意以下几点:

  1. 不建议使用入参是 double 的构造器, 因为它仍然存在精度问题.
  2. 强烈建议使用入参是 string 的构造器, 或者使用静态方法 valueOf 创建对象, 才能保证精度.
  3. 使用 BigDecimal除法 时, 必须要使用 divide(BigDecimal divisor, int scale, RoundingMode roundingMode), 防止除不尽造成异常.

如有写的不对或者没写清楚的地方, 欢迎大家拍砖, 千万不要留情😄. 你拍的砖将成为我进步的垫脚石😁😁😁😁, 能看到这里的小伙伴, 必须得感谢一波.

相关推荐
浮午几秒前
腾讯AI应用开发一面实录:13道硬核面试题全解析
人工智能·面试·职场和发展
可乐ea1 分钟前
【Spring Boot + MyBatis|第4篇】MyBatis 动态 SQL:if、where、foreach 使用详解
java·spring boot·后端·sql·mybatis
記億揺晃着的那天8 分钟前
Windows 通过 Java 获取可用端口的一个坑:Hyper-V 保留端口导致 UDP 绑定失败
java·windows·udp
组合缺一9 分钟前
SolonCode(编码智能体)支持鸿蒙 PC
java·华为·ai·ai编程·harmonyos·solon·soloncode
小bo波11 分钟前
用匿名内部类优雅地计算方法执行时间
java·设计模式·性能测试·模板方法模式·lambda·代码优化·匿名内部类
拾光师14 分钟前
Java AIO 详解:异步非阻塞 IO 的实现与实践
后端
折哥的程序人生 · 物流技术专研15 分钟前
Tomcat 严重警告:JDBC 驱动未注销 + 工作线程泄漏 —— 原因、影响与彻底修复(生产级终极指南)
java·运维·数据库·mysql·oracle·tomcat
一个儒雅随和的男子18 分钟前
sentinel底层原理剖析以及实战优化
java·网络·sentinel
伊灵eLing24 分钟前
GoLang 语言基础
开发语言·后端·golang
两年半的个人练习生^_^24 分钟前
JMM 进阶:彻底理解 synchronized 实现原理
java·开发语言