java判断bigdecimal是否为空_为什么阿里巴巴禁止使用BigDecimal的equals方法做等值比较?

BigDecimal,相信对于很多人来说都不陌生,很多人都知道他的用法,这是一种java.math包中提供的一种可以用来进行精确运算的类型。


很多人都知道,在进行金额表示、金额计算等场景,不能使用double、float等类型,而是要使用对精度支持的更好的BigDecimal。


所以,很多支付、电商、金融等业务中,BigDecimal的使用非常频繁。而且不得不说这是一个非常好用的类,其内部自带了很多方法,如加,减,乘,除等运算方法都是可以直接调用的。


除了需要用BigDecimal表示数字和进行数字运算以外,代码中还经常需要对于数字进行相等判断。


关于这个知识点,在最新版的《阿里巴巴Java开发手册》中也有说明:

c9f4053c3df5ba8aecfb475fe09b783a.png

这背后的思考是什么呢?

我在之前的CodeReview中,看到过以下这样的低级错误:

if(bigDecimal == bigDecimal1){
    // 两个数相等
}

这种错误,相信聪明的读者一眼就可以看出问题,因为BigDecimal是对象,所以不能用==来判断两个数字的值是否相等。

以上这种问题,在有一定的经验之后,还是可以避免的,但是聪明的读者,看一下以下这行代码,你觉得他有问题吗:

if(bigDecimal.equals(bigDecimal1)){
    // 两个数相等
}

可以明确的告诉大家,以上这种写法,可能得到的结果和你预想的不一样!

先来做个实验,运行以下代码:

BigDecimal bigDecimal = new BigDecimal(1);
BigDecimal bigDecimal1 = new BigDecimal(1);
System.out.println(bigDecimal.equals(bigDecimal1));


BigDecimal bigDecimal2 = new BigDecimal(1);
BigDecimal bigDecimal3 = new BigDecimal(1.0);
System.out.println(bigDecimal2.equals(bigDecimal3));


BigDecimal bigDecimal4 = new BigDecimal("1");
BigDecimal bigDecimal5 = new BigDecimal("1.0");
System.out.println(bigDecimal4.equals(bigDecimal5));

以上代码,输出结果为:

true
true
false

BigDecimal的equals原理
通过以上代码示例,我们发现,在使用BigDecimal的equals方法对1和1.0进行比较的时候,有的时候是true(当使用int、double定义BigDecimal时),有的时候是false(当使用String定义BigDecimal时)。


那么,为什么会出现这样的情况呢,我们先来看下BigDecimal的equals方法。


在BigDecimal的JavaDoc中其实已经解释了其中原因:

Compares this  BigDecimal with the specified Object for equality.  Unlike compareTo, this method considers two BigDecimal objects equal only if they are equal in value and scale (thus 2.0 is not equal to 2.00 when compared by  this method)

大概意思就是,equals方法和compareTo并不一样,equals方法会比较两部分内容,分别是值(value)和精度(scale)

对应的代码如下:

e88918b23763bbb77164731a1eac0b99.png

所以,我们以上代码定义出来的两个BigDecimal对象(bigDecimal4和bigDecimal5)的精度是不一样的,所以使用equals比较的结果就是false了。


尝试着对代码进行debug,在debug的过程中我们也可以看到bigDecimal4的精度时0,而bigDecimal5的精度是1。

f5caf4685aa953d484952ef8da1ab6b7.png

到这里,我们大概解释清楚了,之所以equals比较bigDecimal4和bigDecimal5的结果是false,是因为精度不同。


那么,为什么精度不同呢?为什么bigDecimal2和bigDecimal3的精度是一样的(当使用int、double定义BigDecimal时),而bigDecimal4和bigDecimal5却不一样(当使用String定义BigDecimal时)呢?

为什么精度不同


这个就涉及到BigDecimal的精度问题了,这个问题其实是比较复杂的,由于不是本文的重点,这里面就简单介绍一下吧。大家感兴趣的话,后面单独讲。


首先,BigDecimal一共有以下4个构造方法:

BigDecimal(int)
BigDecimal(double) 
BigDecimal(long) 
BigDecimal(String)

以上四个方法,创建出来的的BigDecimal的精度是不同的。

BigDecimal(long) 和BigDecimal(int)

首先,最 简 单的 就是BigDecimal(long) 和BigDecimal(int), 因为是 整数,所 以 精 度就 是0

public BigDecimal(int val) {
    this.intCompact = val;
    this.scale = 0;
    this.intVal = null;
}

public BigDecimal(long val) {
    this.intCompact = val;
    this.intVal = (val == INFLATED) ? INFLATED_BIGINT : null;
    this.scale = 0;
}

BigDecimal(double)


而对于BigDecimal(double) ,当我们使用new BigDecimal(0.1)创建一个BigDecimal 的时候,其实创建出来的值并不是整好等于0.1的,而是0.1000000000000000055511151231257827021181583404541015625 。这是因为doule自身表示的只是一个近似值。


那么,无论我们使用new BigDecimal(0.1)还是new BigDecimal(0.10)定义,他的近似值都是0.1000000000000000055511151231257827021181583404541015625这个,那么他的精度就是这个数字的位数,即55。

f94c532c8c0abdf00194f1c79a828a18.png

其他的浮点数也同样的道理。对于new BigDecimal(1.0)这样的形式来说,因为他本质上也是个整数,所以他创建出来的数字的精度就是0。


所以,因为BigDecimal(1.0)和BigDecimal(1.00)的精度是一样的,所以在使用equals方法比较的时候,得到的结果就是true。

BigDecimal(string)


而对于BigDecimal(double) ,当我们使用new BigDecimal("0.1")创建一个BigDecimal 的时候,其实创建出来的值正好就是等于0.1的。那么他的精度也就是1。


如果使用new BigDecimal("0.10000"),那么创建出来的数就是0.10000,精度也就是5。


所以,因为BigDecimal("1.0")和BigDecimal("1.00")的精度不一样,所以在使用equals方法比较的时候,得到的结果就是false。如何比较BigDecimal


前面,我们解释了BigDecimal的equals方法,其实不只是会比较数字的值,还会对其精度进行比较。


所以,当我们使用equals方法判断判断两个数是否相等的时候,是极其严格的。


那么,如果我们只想判断两个BigDecimal的值是否相等,那么该如何判断呢?


BigDecimal中提供了compareTo方法,这个方法就可以只比较两个数字的值,如果两个数相等,则返回0。

 BigDecimal bigDecimal4 = new BigDecimal("1");
    BigDecimal bigDecimal5 = new BigDecimal("1.0000");
    System.out.println(bigDecimal4.compareTo(bigDecimal5));

以上代码,输出结果:

0

其源码如下:

594a71d616da4b6778499fbad756c488.png

总结


BigDecimal是一个非常好用的表示高精度数字的类,其中提供了很多丰富的方法。


但是,他的equals方法使用的时候需要谨慎,因为他在比较的时候,不仅比较两个数字的值,还会比较他们的精度,只要这两个因素有一个是不相等的,那么结果也是false、


如果读者想要对两个BigDecimal的数值进行比较的话,可以使用compareTo方法。

原文:为什么阿里巴巴禁止使用BigDecimal的equals方法做等值比较?
平台:掘金
作者: HollisChuang
  • 5
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 可以使用BigDecimal的compareTo方法判断是否为,如果为则为。具体代码如下: BigDecimal bd = new BigDecimal(""); if (bd.compareTo(BigDecimal.ZERO) == ) { System.out.println("BigDecimal"); } else { System.out.println("BigDecimal不为"); } ### 回答2: 在Java中,BigDecimal是一个非常常用的类,通常用于处理浮点数和非常大或非常小的数值,例如货币计算等。因此,我们经常需要判断一个BigDecimal对象是否或为null。 判断一个BigDecimal是否或为null,可以使用以下几种方法: 1. 判断是否为null: 可以使用Java中的null关键字来判断一个BigDecimal对象是否为null。例如: ``` BigDecimal decimal = null; if (decimal == null) { System.out.println("decimal is null"); } ``` 2. 判断是否为0: 使用compareTo方法BigDecimal与0进行比较,如果结果为0,则说明它是一个值。例如: ``` BigDecimal decimal = new BigDecimal("0"); if (decimal.compareTo(BigDecimal.ZERO) == 0) { System.out.println("decimal is empty"); } ``` 3. 判断是否值: 调用BigDecimal类的equals方法判断是否值。例如: ``` BigDecimal decimal = null; if (decimal != null && decimal.equals(BigDecimal.ZERO)) { System.out.println("decimal is empty"); } ``` 总之,在Java判断一个BigDecimal对象是否或为null,需要使用以上三种方法之一。方法的选择应该根据具体情况来决定,选择最适合自己的方法可以使代码更加简洁和易于理解。 ### 回答3: Java中,判断一个BigDecimal对象是否,可以通过以下几个方法: 1. 使用null值判断: 在Java中,BigDecimal对象的默认值为null。因此,我们可以使用null值来判断一个BigDecimal对象是否。例如: BigDecimal num = null; if (num == null) { System.out.println("BigDecimal对象为"); } 2. 使用equals()方法判断BigDecimal类实现了equals()方法,因此我们可以使用方法比较两个BigDecimal对象是否相等。如果一个BigDecimal对象为null,即使它的值为0,也无法调用该方法。因此,在使用equals()方法比较两个BigDecimal对象之前,需要先进行判断,否则会发生指针异常。例如: BigDecimal num = new BigDecimal("0.0"); if (num == null || num.equals(BigDecimal.ZERO)) { System.out.println("BigDecimal对象为"); } 3. 使用compareTo()方法判断BigDecimal类也实现了compareTo()方法,该方法会根据两个BigDecimal对象的大小关系返回一个整数值。当两个BigDecimal对象相等时,返回0;当第一个对象小于第二个对象时,返回小于0的值;当第一个对象大于第二个对象时,返回大于0的值。因此,我们可以根据compareTo()方法的返回值来判断一个BigDecimal对象是否。例如: BigDecimal num = null; if (num == null || num.compareTo(BigDecimal.ZERO) == 0) { System.out.println("BigDecimal对象为"); } 在以上三种方法中,建议优先使用null值判断方法,因为它的语法简单、清晰易懂,且可以避免指针异常的发生。同时,如果一个BigDecimal对象为null,使用compareTo()或equals()方法比较会返回指针异常或false,可能会导致代码出现难以识别的错误。因此,在判断BigDecimal对象是否时,需要谨慎使用比较方法

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值