在JDK版本为1.8的情况运行下面的代码,会发现很神奇的情况见运行结果)。
看如下代码:
packagecom.longge.mytest;importjava.math.BigDecimal;importjava.math.RoundingMode;importjava.text.DecimalFormat;importorg.junit.Test;public classTestDecimal {
@Testpublic voidtest) {
BigDecimal a= new BigDecimal0.075);
BigDecimal a1= new BigDecimal10.075);
BigDecimal a2= new BigDecimal100.075);
BigDecimal b= new BigDecimal0.074);
BigDecimal b1= new BigDecimal10.074);
BigDecimal b2= new BigDecimal100.074);
BigDecimal c= new BigDecimal“0.075”);
BigDecimal c1= new BigDecimal“10.075”);
BigDecimal c2= new BigDecimal“100.075”);
DecimalFormat df= new DecimalFormat“#0.00”);
BigDecimal one=BigDecimal.ONE;//0.07
System.out.printlna.divideone, 2, RoundingMode.HALF_UP).doubleValue));//0.08
System.out.printlna.divideone).addnew BigDecimal“0.0000000001”)).setScale2, RoundingMode.HALF_UP).doubleValue));//0.07
System.out.printlnb.divideone, 2, RoundingMode.HALF_UP).doubleValue));//0.07
System.out.printlnb.divideone).addnew BigDecimal“0.0000000001”)).setScale2, RoundingMode.HALF_UP).doubleValue));//0.07
System.out.printlndf.format0.074));//0.07
System.out.printlndf.format0.075));//0.08
System.out.printlnc.divideone, 2, RoundingMode.HALF_UP).doubleValue));
System.out.println“—————————-“);//10.07
System.out.printlna1.divideone, 2, RoundingMode.HALF_UP).doubleValue));//10.08
System.out.printlna1.divideone).addnew BigDecimal“0.0000000001”)).setScale2, RoundingMode.HALF_UP).doubleValue));//10.7
System.out.printlnb1.divideone, 2, RoundingMode.HALF_UP).doubleValue));//10.07
System.out.printlnb1.divideone).addnew BigDecimal“0.0000000001”)).setScale2, RoundingMode.HALF_UP).doubleValue));//10.07
System.out.printlndf.format10.074));//10.07
System.out.printlndf.format10.075));//10.08
System.out.printlnc1.divideone, 2, RoundingMode.HALF_UP).doubleValue));
System.out.println“—————————-“);//100.08
System.out.printlna2.divideone, 2, RoundingMode.HALF_UP).doubleValue));//100.08
System.out.printlna2.divideone).addnew BigDecimal“0.0000000001”)).setScale2, RoundingMode.HALF_UP).doubleValue));//100.07
System.out.printlnb2.divideone, 2, RoundingMode.HALF_UP).doubleValue));//100.07
System.out.printlnb2.divideone).addnew BigDecimal“0.0000000001”)).setScale2, RoundingMode.HALF_UP).doubleValue));//100.07
System.out.printlndf.format100.074));//100.08
System.out.printlndf.format100.075));//100.08
System.out.printlnc2.divideone, 2, RoundingMode.HALF_UP).doubleValue));
}
}
运行结果如下:
0.07
0.08
0.07
0.07
0.07
0.07
0.08
—————————-
10.07
10.08
10.07
10.07
10.07
10.07
10.08
—————————-
100.08
100.08
100.07
100.07
100.07
100.08
100.08
其中可选模式有:
1、 ROUND_UP:远离零方向舍入。向绝对值最大的方向舍入,只要舍弃位非0即进位。
2、 ROUND_DOWN:趋向零方向舍入。向绝对值最小的方向输入,所有的位都要舍弃,不存在进位情况。
3、 ROUND_CEILING:向正无穷方向舍入。向正最大方向靠拢。若是正数,舍入行为类似于ROUND_UP,若为负数,舍入行为类似于ROUND_DOWN。Math.round)方法就是使用的此模式。
4、 ROUND_FLOOR:向负无穷方向舍入。向负无穷方向靠拢。若是正数,舍入行为类似于ROUND_DOWN;若为负数,舍入行为类似于ROUND_UP。
5、 HALF_UP:最近数字舍入5进)。这是我们最经典的四舍五入。
6、 HALF_DOWN:最近数字舍入5舍)。在这里5是要舍弃的。
7、 HAIL_EVEN:银行家舍入法。
查看JDK的官方说明:
所以在做对精度要求高的计算时,要将double/float转成字符串后再转成BigDecimal进行计算