Java中BigDecimal与0比较的一个坑实战记录

目录
  • 前言
  • 问题解决
  • 问题讲解
  • 附:Java BigDecimal中一些其他的坑
  • 总结

前言

在之前做的一个项目中,为了保证BigDecimal在除数 divide时,如果被除数为0,出现java.lang.ArithmeticException: / by zero 报错问题,写了一个对比。具体代码如下:

public static BigDecimal getScale(BigDecimal data1,BigDecimal data2,Integer scale){
    if(Objects.isNull(data1) || BigDecimal.ZERO.equals(data1)
        || Objects.isNull(data2) || BigDecimal.ZERO.equals(data2)){
        return BigDecimal.ZERO;
    }
    return data1.divide(data2,scale,BigDecimal.ROUND_HALF_UP);
}

咋一看,的确没毛病。

当被除数为0时,直接由判断返回,并不会进入下面的计算流程。

但,却忽略了一个问题,如果数据是从数据库中查询,数据库中设定的数的格式是0.00时,此时完美绕开了判断。

原因就在于:

new BigDecimal("0.00").equals(BigDecimal.ZERO) 为 false。

问题解决

其实这类问题,很好解决,在BigDecimal源码中,有一个compareTo的方法

简单写个demo,验证效果:

System.out.println(new BigDecimal("0.0").compareTo(BigDecimal.ZERO));
System.out.println(new BigDecimal("0.00").compareTo(BigDecimal.ZERO));
System.out.println(new BigDecimal("0.0000000").compareTo(BigDecimal.ZERO));

System.out.println(new BigDecimal("0").compareTo(BigDecimal.ZERO));

System.out.println(new BigDecimal("0.1").compareTo(BigDecimal.ZERO));

分别打印上面的执行结果,可以发现

不管是什么格式的0,与0相等则返回0;比0大则返回1;比0小则返回-1。

问题讲解

通过compareTo就能解决不同格式的零值,与0使用equals相比为false的问题。

附:Java BigDecimal中一些其他的坑

1.当精度不准确时汇报异常:

代码如下:

BigDecimal a = new BigDecimal("1.2222");
System.out.println(a.setScale(2));

异常如下:

Exception in thread "main" java.lang.ArithmeticException: Rounding necessary
    at java.math.BigDecimal.commonNeedIncrement(BigDecimal.java:4148)
    at java.math.BigDecimal.needIncrement(BigDecimal.java:4204)
    at java.math.BigDecimal.divideAndRound(BigDecimal.java:4112)
    at java.math.BigDecimal.setScale(BigDecimal.java:2452)
    at java.math.BigDecimal.setScale(BigDecimal.java:2512)
    at com.hj.demo.test.Test01.main(Test01.java:8)

需要加上第二个参数:

BigDecimal a = new BigDecimal("1.2222");
System.out.println(a.setScale(2,BigDecimal.ROUND_HALF_UP));

运行结果如下:

1.22

2. BigDecimal做除法时如果出现除不尽(循环小数)时汇报异常:

BigDecimal a = new BigDecimal("1");
BigDecimal b = new BigDecimal("3");

System.out.println("a / b =" + a.divide(b));

异常如下:

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 com.hj.demo.test.Test01.main(Test01.java:10)

此时应该给予divide第二个和第三个参数用来保留小数位数:

BigDecimal a = new BigDecimal("1");
BigDecimal b = new BigDecimal("3");

System.out.println("a / b =" + a.divide(b,2,BigDecimal.ROUND_HALF_UP));

运行结果如下:

a / b =0.33

总结

到此这篇关于Java中BigDecimal与0比较的一个坑的文章就介绍到这了,更多相关Java BigDecimal与0比较内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • java中BigDecimal和0比较的示例代码

    BigDecimal 和 0 比较大小 调用BigDecimal中的compareTo方法, 如: int i = bigDecimal.compareTo(BigDecimal.Zero); i=0:表示bigDecimal的值  等于 0 i=1:表示bigDecimal的值与 大于0 i=-1:表示bigDecimal的值与 小于 0 参考案例 BigDecimal num= new BigDecimal("18"); int i=num.compareTo(BigDecimal

  • Java中BigDecimal与0比较的一个坑实战记录

    目录 前言 问题解决 问题讲解 附:Java BigDecimal中一些其他的坑 总结 前言 在之前做的一个项目中,为了保证BigDecimal在除数 divide时,如果被除数为0,出现java.lang.ArithmeticException: / by zero 报错问题,写了一个对比.具体代码如下: public static BigDecimal getScale(BigDecimal data1,BigDecimal data2,Integer scale){ if(Objects.

  • Java中BigDecimal精度和相等比较的坑

    为什么要有BigDecimal ,他是干什么的 float和double类型的主要设计目标是为了科学计算和工程计算.他们执行二进制浮点运算,这是为了在广域数值范围上提供较为精确的快速近似计算而精心设计的.然而,它们没有提供完全精确的结果,所以不应该被用于要求精确结果的场合.但是,商业计算往往要求结果精确,这时候就要使用BigDecimal啦. 什么是BigDecimal BigDecimal 由任意精度的整数非标度值 和32 位的整数标度 (scale) 组成.如果为零或正数,则标度是小数点后的

  • 一个Java中BigDecimal的问题记录

    题主今天在用 BigDecimal 进行计算的时候,遇到一个问题,那就是不管怎么计算,最后得到的值都没有变化,这里题主就有些疑惑了,用的也是推荐的API,先看代码: BigDecimal totalPay = new BigDecimal("0"); BigDecimal totalIncomeOfMaster = new BigDecimal("0"); BigDecimal totalIncomeOfLB = new BigDecimal("0&quo

  • 详谈Java中BigDecimal的一个除法异常

    如下所示: java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result. 说明遇到除不尽的情况了,需要指定商的小数精度和舍入模式. 比如: a=b.divide(c,2,RoundingMode.HALF_UP); 以上这篇详谈Java中BigDecimal的一个除法异常就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们.

  • Java中BigDecimal的基本运算(详解)

    BigDecimal一共有4个够造方法,让来看看其中比较常用的两种用法: 第一种:BigDecimal(double val) Translates a double into a BigDecimal. 第二种:BigDecimal(String val) Translates the String repre sentation of a BigDecimal into a BigDecimal. 使用BigDecimal要用String来够造,要做一个加法运算,需要先将两个浮点数转为Str

  • 浅谈java中BigDecimal的equals与compareTo的区别

    这两天在处理支付金额校验的时候出现了点问题,有个金额比较我用了BigDecimal的equals方法来比较两个金额是否相等,结果导致金额比较出现错误(比如3.0与3.00的比较等). [注:以下所讲都是以sun jdk 1.4.2版本为例,其他版本实现未必一致,请忽略] 首先看一下BigDecimal的equals方法: public boolean equals(Object x){ if (!(x instanceof BigDecimal)) return false; BigDecima

  • Java中BigDecimal类的简单用法

    本文实例讲述了Java中BigDecimal类的简单用法,是Java程序设计中非常实用的技巧,分享给大家供大家参考.具体用法分析如下: 一般来说,一提到Java里面的商业计算,我们都知道不能用float和double,因为他们无法进行精确计算.但是Java的设计者给编程人员提供了一个很有用的类BigDecimal,他可以完善float和double类无法进行精确计算的缺憾.BigDecimal类位于java.maths类包下.首先我们来看下如何构造一个BigDecimal对象.它的构造函数很多,

  • 浅谈java中BigDecimal类的简单用法

    一.BigDecimal概述 ​ Java在java.math包中提供的API类BigDecimal,用来对超过16位有效位的数进行精确的运算.双精度浮点型变量double可以处理16位有效数,但在实际应用中,可能需要对更大或者更小的数进行运算和处理.一般情况下,对于那些不需要准确计算精度的数字,我们可以直接使用Float和Double处理,但是Double.valueOf(String) 和Float.valueOf(String)会丢失精度.所以开发中,如果我们需要精确计算的结果,则必须使用

  • 详解java中BigDecimal精度问题

    一.背景 在实际开发中,对于 不需要任何准确计算精度的属性可以直接使用float或double,但是如果需要精确计算结果,则必须使用BigDecimal,例如价格.质量. 为什么这么说,主要有两点 1.double计算会有精度丢失问题 2.在除法运算时,BigDecimal提供了丰富的取舍规则.(double虽然可以通过NumberFormat进行四舍五入,但是NumberFormat是线程不安全的) 对于精度问题我们可以看下实际的例子 public static void main(Strin

随机推荐