美文网首页Java基础
BigDecimal踩坑集锦

BigDecimal踩坑集锦

作者: 莫问以 | 来源:发表于2019-06-20 17:00 被阅读3次

    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));
    
    保留小数位数精度错误.png

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

    相关文章

      网友评论

        本文标题:BigDecimal踩坑集锦

        本文链接:https://www.haomeiwen.com/subject/zqokyqtx.html