【问题记录】BigDecimal 的常见坑
1 前言
我们现在公司的项目,涉及到金额、小数的都会用 BigDecimal 来操作,比如订单的各种金额,支付的各种金额等都会用它,那么你们平时有没有遇到的坑呢?我们这节来看看它常见的坑。
2 常见坑
2.1 new BigDecimal()还是BigDecimal#valueOf()?
先看下面这段代码:
BigDecimal bd1 = new BigDecimal(0.01); BigDecimal bd2 = BigDecimal.valueOf(0.01); System.out.println("bd1 = " + bd1); System.out.println("bd2 = " + bd2);
输出到控制台的结果是:
bd1 = 0.01000000000000000020816681711721685132943093776702880859375
bd2 = 0.01
造成这种差异的原因是0.1这个数字计算机是无法精确表示的,送给BigDecimal
的时候就已经丢精度了,而BigDecimal#valueOf
的实现却完全不同:
public static BigDecimal valueOf(double val) { // Reminder: a zero double returns '0.0', so we cannot fastpath // to use the constant ZERO. This might be important enough to // justify a factory approach, a cache, or a few private // constants, later. return new BigDecimal(Double.toString(val)); }
它使用了浮点数相应的字符串来构造BigDecimal
对象,因此避免了精度问题。所以大家要尽量要使用字符串而不是浮点数去构造BigDecimal
对象,如果实在不行,就使用BigDecimal#valueOf()
方法吧。
2.2 等值比较
BigDecimal bd1 = new BigDecimal("1.0"); BigDecimal bd2 = new BigDecimal("1.00"); System.out.println(bd1.equals(bd2)); System.out.println(bd1.compareTo(bd2));
控制台的输出将会是:
false
0
究其原因是,BigDecimal
中equals
方法的实现会比较两个数字的精度,而compareTo
方法则只会比较数值的大小。
2.3 BigDecimal并不代表无限精度
先看这段代码:
BigDecimal a = new BigDecimal("1.0"); BigDecimal b = new BigDecimal("3.0"); a.divide(b) // results in the following exception.
结果会抛出异常:
java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.
关于这个异常,Oracle的官方文档有具体说明If the quotient has a nonterminating decimal expansion and the operation is specified to return an exact result, an ArithmeticException is thrown. Otherwise, the exact result of the division is returned, as done for other operations.大意是,如果除法的商的结果是一个无限小数但是我们期望返回精确的结果,那程序就会抛出异常。回到我们的这个例子,我们需要告诉JVM
我们不需要返回精确的结果就好了
BigDecimal a = new BigDecimal("1.0"); BigDecimal b = new BigDecimal("3.0"); a.divide(b, 2, RoundingMode.HALF_UP)// 0.33
这个还是比较常见的,对于一些经验少的新手来说很容易碰到,所以除法一定要设置精度,否则测试没发现,到了生产环境碰到了就嗝屁了。
2.4 BigDecimal转回String要小心
先看这段代码:
BigDecimal d = BigDecimal.valueOf(12334535345456700.12345634534534578901); String out = d.toString(); // Or perform any formatting that needs to be done System.out.println(out); // 1.23345353454567E+16
可以看到结果已经被转换成了科学计数法,可能这个并不是预期的结果BigDecimal
有三个方法可以转为相应的字符串类型,切记不要用错:
String toString(); // 有必要时使用科学计数法 String toPlainString(); // 不使用科学计数法 String toEngineeringString(); // 工程计算中经常使用的记录数字的方法,与科学计数法类似,但要求10的幂必须是3的倍数
2.5 执行顺序不能调换(乘法交换律失效)
乘法满足交换律是一个常识,但是在计算机的世界里,会出现不满足乘法交换律的情况:
BigDecimal a = BigDecimal.valueOf(1.0); BigDecimal b = BigDecimal.valueOf(3.0); BigDecimal c = BigDecimal.valueOf(3.0); System.out.println(a.divide(b, 2, RoundingMode.HALF_UP).multiply(c)); // 0.990 System.out.println(a.multiply(c).divide(b, 2, RoundingMode.HALF_UP)); // 1.00
别小看这这0.01的差别,在汇金领域,会产生非常大的金额差异。
3 拓展
关于金额计算,很多业务团队会基于BigDecimal
再封装一个Money
类,其实我们直接可以用一个半官方的Money
类:JSR 354 ,虽然没能在Java 9
中成为Java
标准,很有可能集成到后续的Java
版本中成为官方库。
<dependency> <groupId>org.javamoney</groupId> <artifactId>moneta</artifactId> <version>1.1</version> </dependency>
新建 Money
类:
CurrencyUnit cny = Monetary.getCurrency("CNY"); Money money = Money.of(1.0, cny); // 或者 Money money = Money.of(1.0, "CNY"); //System.out.println(money);
金额运算:
CurrencyUnit cny = Monetary.getCurrency("CNY"); Money oneYuan = Money.of(1.0, cny); Money threeYuan = oneYuan.add(Money.of(2.0, "CNY")); //CNY 3 Money tenYuan = oneYuan.multiply(10); // CNY 10 Money fiveFen = oneYuan.divide(2); //CNY 0.5
比较相等:
Money fiveFen = Money.of(0.5, "CNY"); //CNY 0.5 Money anotherFiveFen = Money.of(0.50, "CNY"); // CNY 0.50 System.out.println(fiveFen.equals(anotherFiveFen)); // true
可以看到,这个类对金额做了显性的抽象,增加了金额的单位,也避免了直接使用BigDecimal
的一些坑。
4 小结
好啦,关于 BigDecimal 的坑暂时记录这些哈,尤其是除法要设置精度哈,有理解不对的地方欢迎指正哈。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· Obsidian + DeepSeek:免费 AI 助力你的知识管理,让你的笔记飞起来!
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了