BigDecimal在Java编程中,非常实用,在算钱的时候,哪怕精度丢了一丢丢都让人头疼,所以一般我们不采用float类型和double来做货币计算。
System.out.println(0.05 + 0.01); //0.060000000000000005
System.out.println(1.0 - 0.66); //0.33999999999999997
System.out.println(3.1415088 * 100); //314.15088
System.out.println(314.15088 / 100); //3.1415087999999995
比如说我们简单计算1-0.66的问题,精度实在不准,所以Java专门提供了一个类java.math来做大数字(超过16位有效位)运算的操作类,即 java.math.BinInteger 类和 java.math.BigDecimal 类,用于高精度计算。其中 BigInteger 类是针对整数处理, BigDecimal 类则是针对大小数的处理。
坑一:new BigDecimal()对象:
BigDecimal strTwo = new BigDecimal(0.1);
//0.1000000000000000055511151231257827021181583404541015625
BigDecimal str = new BigDecimal("0.1"); //0.1
BigDecimal(double d)表示的不是0.1 而是0.1000000000000000055511151231257827021181583404541015625,这是因为0.1无法准确的表示为double,所以BigDecimal(double d)不推荐使用,也不允许使用,而应该应该使用 new BigDecimal(String str);
不推荐使用BigDecimal(double val)构造器,是因为使用该构造器时有不可预知性,当程序使用new BigDecimal(0.1)创建一个BigDecimal对象时,它的值并不是0.1,实际上是一个近似0.1的数。
如果必须使用double浮点数作为BigDecimal构造器的参数时,可以通过BigDecimal.valueOf(double value)静态方法来创建对象。
BigDecimal s2 = BigDecimal.valueOf(0.998); //0.998
坑二:BigDecimal 不可变:
BigDecimal 和String 一样具有对象不可变行,一旦赋值就不会再变,即便做加减乘除运算:
BigDecimal count = new BigDecimal("3.1415");
count.add(new BigDecimal("0.1645"));
System.out.println("count:" + count); //count:3.1415
// 并非在count基础上做运算
System.out.println("result:" + count.add(new BigDecimal("0.1645"))); //result:3.3060
这个也是常踩的坑之一,BigDecimal运算方法如下:
public BigDecimal add(BigDecimal value); //加法
public BigDecimal subtract(BigDecimal value); //减法
public BigDecimal multiply(BigDecimal value); //乘法
public BigDecimal divide(BigDecimal value); //除法
针对除法://商,几位小数,舍取模式
public BigDecimal divide(BigDecimal value,int scale, int roundingMode);
坑三:保留小数位数:
BigDecimal保留小数位数,主要用setScale
方法:
BigDecimal a = new BigDecimal("1.6666");
System.out.println(a.setScale(2));
这个坑很容易就踩了,setScale
方法构造函数有下:
public BigDecimal setScale(int newScale)
public BigDecimal setScale(int newScale, int roundingMode)
public BigDecimal setScale(int newScale, RoundingMode roundingMode)//同上
第2个参数说明:
ROUND_CEILING //向正无穷方向舍入
ROUND_DOWN //向零方向舍入
ROUND_FLOOR //向负无穷方向舍入
ROUND_HALF_DOWN //向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,向下舍入,例如1.55 保留一位小数结果为1.5
ROUND_HALF_EVEN //向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,如果保留位数是奇数,使用ROUND_HALF_UP,如果是偶数,使用ROUND_HALF_DOWN
ROUND_HALF_UP //向(距离)最近的一边舍入,除非两边(的距离)是相等,如果是这样,向上舍入, 1.55保留一位小数结果为1.6
ROUND_UNNECESSARY //计算结果是精确的,不需要舍入模式
ROUND_UP //向远离0的方向舍入
BigDecimal a = new BigDecimal("1.6666");
// System.out.println(a.setScale(2)); //精度错误
System.out.println(a.setScale(2,BigDecimal.ROUND_HALF_UP)); // 1.67
BigDecimal .devide()
坑和这个类似:
BigDecimal a = new BigDecimal("1");
BigDecimal b = new BigDecimal("3");
//System.out.println("a / b =" + a.divide(b)); //错误
System.out.println("a / b =" + a.divide(b, 2, BigDecimal.ROUND_HALF_UP)); //a / b =0.33
Exception in thread "main" java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.
at java.math.BigDecimal.divide(BigDecimal.java:1690)
at xf.TestNull.main(TestNull.java:19)