关于四舍五入,大家应该都知道,如0.4四舍为0,0.6五入到1,按照正常的逻辑习惯,0.5是属于五入范围的,但在Java某些方法并不是这样的哦!
DecimalFormat df=new DecimalFormat("0.00");
System.out.println(df.format(0.045));
System.out.println(df.format(0.044));
System.out.println(df.format(0.046));
实际输出结果是:0.04 0.04 0.05
在进行某些统计分析的时候,保留两位小数是很普遍的需求,但是这样的四舍五入结果你能接受吗?或者说客户能接受吗?这样的结果与其说是精度问题,不如说是严重代码错误问题!
那为什么会这样呢?
这是因为这里的四舍五入默认使用了银行家舍入(Banker's Round)算法。
这是个啥子算法哟?
如果你是个银行家,你的银行需要进行结算利息,银行利息它就算到小数点后两位,如100.00。按照常规的四舍五入规则,会产生一种神奇的亏损。
当尾数为0,1,2,3,4的时候,会执行舍去规则。这样100.004,支付100.00就能赚到0.004。这样赚的情况为+0.000,+0.001,+0.002,+0.003,+0.004
当尾数为5,6,7,8,9的时候,会执行入位规则。这样100.005,支付100.01就会亏损0.005。这样亏的情况为-0.005,-0.004,-0.003,-0.002,-0.001
把所有情况都加一起,结果为-0.005。平均每10笔交易亏损0.005,这样算起来每笔交易就亏0.0005。这样玩就不好玩啦!一家大型银行如果使用这种常规的四舍五入算法,不是就平白在利息上少赚了一大笔钱吗?
So,美国银行家发现了这个问题。少赚了钱可不行,搞一个新的四舍五入算法!不能吃亏!
于是银行家舍入算法就出场了!
它的规则可以简单概述为:四舍六入五考虑,五后非零就进一,五后为零看奇偶,五前为偶应舍去,五前为奇要进一
如果要使用经典的四舍五入算法,可以使用不同的计算模式
DecimalFormat df=new DecimalFormat("0.00");
df.setRoundingMode(RoundingMode.HALF_UP);
System.out.println(df.format(0.045));
System.out.println(df.format(0.044));
System.out.println(df.format(0.046));
输出结果为:0.05 0.04 0.05
这里说明其他几种模式:
- ROUND_UP: 远离零方向舍入。
- 向远离0的方向舍入,也就是说,向绝对值最大的方向舍入,只要舍弃位非0即进位。
- ROUND_DOWN:趋向零方向舍入。
- 向0方向靠拢,也就是说,向绝对值最小的方向输入,注意:所有的位都舍弃,不存在进位情况。
- ROUND_CEILING:向正无穷方向舍入。
- 向正最大方向靠拢,如果是正数,舍入行为类似于ROUND_UP;如果为负数,则舍入行为类似于ROUND_DOWN。注意:Math.round方法使用的即为此模式。
- ROUND_FLOOR:向负无穷方向舍入。
- 向负无穷方向靠拢,如果是正数,则舍入行为类似于 ROUND_DOWN;如果是负数,则舍入行为类似于 ROUND_UP。
- HALF_UP: 最近数字舍入(5进)。
- 这就是我们最最经典的四舍五入模式。
- HALF_DOWN:最近数字舍入(5舍)。
- 在四舍五入中,5是进位的,而在HALF_DOWN中却是舍弃不进位。
- HALF_EVEN :银行家舍入算法。