问题描述:
BigDecimal做除法时需要设置精度,不然会报错;今天需要算一个百分比,直接分子乘以100再除以分母,发现所设的四舍五入失效。
eg:
BigDecimal x = BigDecimal.valueOf(3);
BigDecimal y = BigDecimal.valueOf(2);
BigDecimal z = BigDecimal.valueOf(4);
//四舍五入失效
System.out.println(y.multiply(BigDecimal.valueOf(100).divide(x, 2, RoundingMode.HALF_UP)));//66.66
//直接两个数计算正常
System.out.println(BigDecimal.valueOf(200).divide(x, 2, RoundingMode.HALF_UP));//66.67
//正确做法-运算完乘法取出隔开
System.out.println((y.multiply(BigDecimal.valueOf(100)).divide(x, 2, RoundingMode.HALF_UP)));//66.67
//先除后乘精度不对
System.out.println(y.divide(x, 2, RoundingMode.HALF_UP).multiply(BigDecimal.valueOf(100)));//67.00
//并没有先乘除后加减,而是一步步来
System.out.println(x.add(y).multiply(z));//20
总结:
BigDecimal先乘再除连续计算时比较特殊,需要先乘完再做除法,特此记录注意一下,后续深究下原理
拓展:
RoundingMode参数
RoundingMode | 说明 |
---|---|
UP | 正数向右边靠,负数向左边靠 |
DOWN | 跟UP相反,正数向左边靠,负数向右边靠 |
CEILING | 全向右靠,向大靠拢 |
FLOOR | 跟CEILING相反,全向左靠,向小靠拢 |
HALF_UP | 四舍五入 |
HALF_DOWN | 五舍六入 |
HALF_EVEN | 银行家舍入法(四舍六入五考虑,五后非空就进一,五后为空看奇偶,五前为偶应舍去,五前为奇要进一) |
UNNECESSARY | 将舍入到“最近的邻居”,除非两个邻居都等距,在这种情况下,将舍入到偶数邻居 |
CEILING | 需要舍掉的位数不是0,就报异常 ArithmeticException,否则正常 |
银行家舍入法:
银行家舍入法是由IEEE 754标准规定的浮点数取整算法 [1] ,大部分的编程软件都使用的是这种方法。 所谓银行家舍入法,其实质是一种四舍六入五取偶(又称四舍六入五留双)法。其规则是:当舍去位的数值小于5时,直接舍去该位;当舍去位的数值大于等于6时,在舍去该位的同时向前位进一;当舍去位的数值等于5且(5后不为空且非全0)时,在舍去该位的同时向前位进一;当舍去位的数值等于5且(5后为空或全0)时,如果前位数值为奇,则在舍去该位的同时向前位进一,如果前位数值为偶,则直接舍去该位。