BigDecimal使用踩坑记录

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));

BigDecimal使用踩坑记录_第1张图片

这个坑很容易就踩了,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)

你可能感兴趣的:(基础)