又来论坛求助 V 友了,今天在使用 BigDecimal 四舍五入保留一位小数的时候遇到了很神奇的问题。
double num1 = 5.95;
double num2 = 6.95;
double num3 = 7.95;
double num4 = 8.95;
double num5 = 9.95;
double num6 = 5.43;
System.out.println(new BigDecimal(num1).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 6.0
System.out.println(new BigDecimal(num2).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 7.0
System.out.println(new BigDecimal(num3).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 8.0
System.out.println(new BigDecimal(num4).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 8.9
System.out.println(new BigDecimal(num5).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 9.9
System.out.println(new BigDecimal(num6).setScale(1, BigDecimal.ROUND_HALF_UP).doubleValue()); // 5.4
8.95 和 9.95 四舍五入应该是 9.0 和 10.0 ,但是输出不是这样,但是 5.95 、6.95 、7.95 又不是这样,这是为啥捏
1
mango88 2022-08-31 22:09:44 +08:00
System.out.println(new BigDecimal(num1));
System.out.println(new BigDecimal(num4)); 这俩你输出看一下就明白了,浮点数的精度问题 另外构建 BigDecimal 对象推荐用字符串构造 new BigDecimal("5.95") new BigDecimal("8.95") 这样 |
2
dqzcwxb 2022-08-31 22:13:32 +08:00
|
3
mango88 2022-08-31 22:14:27 +08:00
或者 BigDecimal.valueOf(double)
|
4
dqzcwxb 2022-08-31 22:15:20 +08:00
|
5
dramakevinzz OP @mango88 感谢!这种方法真的就解决了,但是还想请教一下 [浮点数的精度问题] 这个是指什么?可以再解释一下嘛?或者可以再细说一下方向,我查一下,谢谢
|
6
dramakevinzz OP @dqzcwxb 感谢,这张图确实看着就比较明显了,我记得好像是浮点数的存储和表象并不一致,导致有可能类型 X.9+Y.1 最后并不是 Z.0 这种情况,看 CSAPP 的时候记得这里好像是和 IEEE 设计有关,我再翻一下书,查查,感谢
|
7
mineralsalt 2022-08-31 23:08:39 +08:00 1
因为浮点的问题, 我现在设计金额相关字段的时候, 全部用 int
|
8
dramakevinzz OP @mineralsalt 金额用 int 吗?这个和日常使用场景怎么契合呢?比如小数,分小数和整数分开存储吗?
|
9
wu67 2022-09-01 00:17:04 +08:00 1
@dramakevinzz 用分为单位, 就不存在小数了
|
10
VeryZero 2022-09-01 08:47:25 +08:00
BigDecimal 源码注释里有解释,建议使用字符串构造
|
11
mineralsalt 2022-09-02 11:51:06 +08:00
@dramakevinzz #8 所有金额都用分进行数据库存储, 前端展示的时候格式化一下就行了, 只要存到数据库, 就是分为单位
|