亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

Java中BigDecimal的equals方法和compareTo方法的區(qū)別詳析

 更新時間:2023年11月28日 11:09:01   作者:liweijackson  
這篇文章主要給大家介紹了關(guān)于Java中BigDecimal的equals方法和compareTo方法區(qū)別的相關(guān)資料,對于BigDecimal的大小比較,用equals方法的話會不僅會比較值的大小,還會比較兩個對象的精確度,而compareTo方法則不會比較精確度,只比較數(shù)值的大小,需要的朋友可以參考下

前言

面試經(jīng)常會問道為什么不能用BigDecimal的equals方法做等值比較?

其實BigDecimal的equals方法和compareTo并不一樣,equals方法會比較兩部分內(nèi)容,分別是值(value)和標度(scale),而對于1.0和1這兩個數(shù)字,他們的值雖然一樣,但是精度是不一樣的,在使用equals比較的時候會返回false。

知識擴展

BigDecimal,相信對于很多人來說都不陌生,很多人都知道他的用法,這是一種java.math包中提供的一種可以用來進行精確運算的類型。很多人都知道,在進行金額表示、金額計算等場景,不能使用double、float等類型,而是要使用對精度支持的更好的BigDecimal。所以,很多支付、電商、金融等業(yè)務(wù)中,BigDecimal的使用非常頻繁。而且不得不說這是一個非常好用的類,其內(nèi)部自帶了很多方法,如加,減,乘,除等運算方法都是可以直接調(diào)用的。

BigDecimal的比較

阿里java開發(fā)手冊也有規(guī)則要求比較不使用equals

【強制】 BigDecimal的等值比較應適應conpareTo()方法,而不是equals()。
說明:equals()方法會比較值和精度(1.0與1.00返回結(jié)果為false),二conpareTo()則會忽略精度

在做代碼評審時見過下面的寫法

if(bigDecimal == bigDecimal1){
	//兩數(shù)相等
}

這種錯誤,相信聰明的讀者一眼就可以看出問題,因為BigDecimal是對象,所以不能用 == 來判斷兩個數(shù)字的值是否相等。

上面代碼的問題,在有一定的經(jīng)驗之后,還是可以避免的,我們可能首先可能會想到使用equals來比較,你覺得他有問題嗎:

if(bigDecimal.equals(bigDecimal1)){
	//兩數(shù)相等
}

以上這種寫法,可能得到的結(jié)果和你預想的完全不一樣!可以做個簡單的測試運行一下代碼

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

輸出結(jié)果為

true
true
false

BigDecimal的equals原理

通過上面的代碼示例,我們發(fā)現(xiàn),在使用BigDecimal的equals方法對1和1.0進行比較的時候,有的時候是true(當使用int、double定義BigDecimal時),有的時候是false(當使用String定義BigDecimal時)。

那么,為什么會出現(xiàn)這樣的情況呢,我們先來看下BigDecimal的equals方法。在BigDecimal的JavaDoc中其實已經(jīng)解釋了其中原因:

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方法會比較兩部分內(nèi)容,分別是值(value)和標度(scale)。
代碼如下:

所以,我們以上代碼定義出來的兩個BigDecimal對象(bigDecimal4和bigDecimal5)的標度是不一樣的,所以使用equals比較的結(jié)果就是false了。

對代碼進行debug,在debug的過程中我們也可以看到bigDecimal4的標度時0,而bigDecimal5的標度是1。到這里,我們大概解釋清楚了,之所以equals比較bigDecimal4和bigDecimal5的結(jié)果是false,是因為標度不同

為什么標度不同

這個就涉及到BigDecimal的標度問題了,這個問題其實是比較復雜的,由于不是本文的重點,不做過多介紹

首先我們可以看到BigDecimal有4個構(gòu)造方法

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

上面4種方法創(chuàng)建出來的BigDecimal標度時不相同的

BigDecimal(long) 和BigDecimal(int)

首先,最簡單的就是BigDecimal(long) 和BigDecimal(int),因為是整數(shù),所以標度就是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)創(chuàng)建一個BigDecimal 的時候,其實創(chuàng)建出來的值并不是整好等于0.1的,而是0.1000000000000000055511151231257827021181583404541015625 。這是因為doule自身表示的只是一個近似值。

那么,無論我們使用new BigDecimal(0.1)還是new BigDecimal(0.10)定義,他的近似值都是0.1000000000000000055511151231257827021181583404541015625這個,那么他的標度就是這個數(shù)字的位數(shù),即55。

在構(gòu)造方法里面打斷點可以驗證

其他的浮點數(shù)也同樣的道理。對于new BigDecimal(1.0)這樣的形式來說,因為他本質(zhì)上也是個整數(shù),所以他創(chuàng)建出來的數(shù)字的標度就是0。所以,因為BigDecimal(1.0)和BigDecimal(1.00)的標度是一樣的,所以在使用equals方法比較的時候,得到的結(jié)果就是true。

BigDecimal(String)

而對于BigDecimal(String) ,當我們使用new BigDecimal(“0.1”)創(chuàng)建一個BigDecimal 的時候,其實創(chuàng)建出來的值正好就是等于0.1的。那么他的標度也就是1。如果使用new BigDecimal(“0.10000”),那么創(chuàng)建出來的數(shù)就是0.10000,標度也就是5。

所以,因為BigDecimal(“1.0”)和BigDecimal(“1.00”)的標度不一樣,所以在使用equals方法比較的時候,得到的結(jié)果就是false。

如何比較BigDecimal

在上面,我們解釋了BigDecimal的equals方法,其實不只是會比較數(shù)字的值,還會對其標度進行比較。所以,當我們使用equals方法判斷判斷兩個數(shù)是否相等的時候,是極其嚴格的。

那么,如果我們只想判斷兩個BigDecimal的值是否相等,那么該如何判斷呢?

BigDecimal中提供了compareTo方法,這個方法就可以只比較兩個數(shù)字的值,如果兩個數(shù)相等,則返回0。

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

以上代碼,輸出結(jié)果:

0

compareTo源碼如下

附:bigdecimal判斷是否為零(compareTo比較大?。?/h2>

一、bigdecimal判斷是否為零

1.判斷Bigdecimal類型是否等于0的方法b.equals(BigDecimal.ZERO);

用equals方法和BigDecimal.ZERO進行比較。

2.上面判斷是否等于零存在的問題:

我們來查看源代碼:

  • Bigdecimal的equals方法不僅僅比較值的大小是否相等,首先比較的是scale(scale是bigdecimal的保留小數(shù)點位數(shù),比如 new Bigdecimal(“1.001”),scale為3),也就是說,不但值得大小要相等,保留位數(shù)也要相等,equals才能返回true。
  • Bigdecimal b = new Bigdecimal(“0”) 和 Bigdecimal c = new Bigdecimal(“0.0”),用equals比較,返回就是false。
  • Bigdecimal.ZERO的scale為0。

所以,用equals方法要注意這一點。

3.我們還可以用一種方式判斷Bigdecimal類型的值是否為0:

b.compareTo(BigDecimal.ZERO)==0,可以比較是否等于0,返回true則等于0,返回false,則不等于0

二、Bigdecimal的compareTo方法使用:

public static void main(String[] args) {
    BigDecimal bnum1, bnum2;

    bnum1 = new BigDecimal("10");
    bnum2 = new BigDecimal("20");

    int res = bnum1.compareTo(bnum2); 

    String str1 = "兩個數(shù)相等";
    String str2 = "第一個數(shù)更大";
    String str3 = "第二個數(shù)更大";

    if( res == 0 )
        System.out.println( str1 );
    else if( res == 1 )
        System.out.println( str2 );
    else if( res == -1 )
        System.out.println( str3 );
    }
}

為什么比較返回值是0,-1和1呢? 我們?nèi)タ纯丛创a!

根據(jù)源碼中的三元運算符可以發(fā)現(xiàn):

  • 情況1. 如果xs等于ys,則返回0。
  • 情況2. 如果xs不等于ys,則會執(zhí)行另外一個三元運算符((xs > ys) ? 1 : -1)

這時候就會比較 xs 和 ys:

  • xs > ys 返回 1,
  • xs < ys 返回 -1。

因此得到結(jié)論!

總結(jié) 

到此這篇關(guān)于Java中BigDecimal的equals方法和compareTo方法區(qū)別的文章就介紹到這了,更多相關(guān)BigDecimal equals和compareTo方法區(qū)別內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • Mybatis分頁插件PageHelper的分頁原理剖析

    Mybatis分頁插件PageHelper的分頁原理剖析

    這篇文章主要介紹了Mybatis分頁插件PageHelper的分頁原理剖析,PageHelper作為一個啟動器,那么就和其他啟動器加載一樣,先讀取spring.factories文件里面配置的類,轉(zhuǎn)成Bean加載本系統(tǒng)中,然后執(zhí)行他的前置后置處理方法,完成初始化,需要的朋友可以參考下
    2023-08-08
  • java操作mongodb之多表聯(lián)查的實現(xiàn)($lookup)

    java操作mongodb之多表聯(lián)查的實現(xiàn)($lookup)

    這篇文章主要介紹了java操作mongodb之多表聯(lián)查的實現(xiàn)($lookup),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-03-03
  • java8?時間日期的使用與格式化示例代碼詳解

    java8?時間日期的使用與格式化示例代碼詳解

    這篇文章主要介紹了java8?時間日期的使用與格式化,本文通過示例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-06-06
  • java利用遞歸實現(xiàn)類別樹示例代碼

    java利用遞歸實現(xiàn)類別樹示例代碼

    這篇文章主要給大家介紹了關(guān)于java利用遞歸實現(xiàn)類別樹的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-12-12
  • Maven pom.xml 添加本地jar包依賴以及打包方法

    Maven pom.xml 添加本地jar包依賴以及打包方法

    這篇文章主要介紹了Maven pom.xml 添加本地jar包依賴以及打包方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2020-09-09
  • Java實現(xiàn)Excel導入導出數(shù)據(jù)庫的方法示例

    Java實現(xiàn)Excel導入導出數(shù)據(jù)庫的方法示例

    這篇文章主要介紹了Java實現(xiàn)Excel導入導出數(shù)據(jù)庫的方法,結(jié)合實例形式分析了java針對Excel的讀寫及數(shù)據(jù)庫操作相關(guān)實現(xiàn)技巧,需要的朋友可以參考下
    2017-08-08
  • SpringBoot+Vue項目部署實現(xiàn)傳統(tǒng)方式

    SpringBoot+Vue項目部署實現(xiàn)傳統(tǒng)方式

    我們在進行前后端分離開發(fā)的時候,一般是將前端項目部署到nginx服務(wù)器上,與后端項目分開部署,這篇文章主要給大家介紹了關(guān)于SpringBoot+Vue項目部署實現(xiàn)傳統(tǒng)方式的相關(guān)資料,需要的朋友可以參考下
    2024-01-01
  • Java中json使用方法_動力節(jié)點Java學院整理

    Java中json使用方法_動力節(jié)點Java學院整理

    JSON(JavaScript Object Notation) 是一種輕量級的數(shù)據(jù)交換格式, json是個非常重要的數(shù)據(jù)結(jié)構(gòu),在web開發(fā)中應用十分廣泛。下面通過本文給大家講解Java中json使用方法,感興趣的朋友一起看看吧
    2017-07-07
  • 基于SpringBoot和Vue3的博客平臺文章詳情與評論功能實現(xiàn)

    基于SpringBoot和Vue3的博客平臺文章詳情與評論功能實現(xiàn)

    在前面的教程中,我們已經(jīng)實現(xiàn)了基于Spring Boot和Vue3的發(fā)布、編輯、刪除文章功能以及文章列表與分頁功能。本教程將引導您實現(xiàn)博客平臺的文章詳情與評論功能,需要的朋友可以參考一下
    2023-04-04
  • 基于ZooKeeper實現(xiàn)隊列源碼

    基于ZooKeeper實現(xiàn)隊列源碼

    這篇文章主要介紹了基于ZooKeeper實現(xiàn)隊列源碼的相關(guān)內(nèi)容,包括其實現(xiàn)原理和應用場景,以及對隊列的簡單介紹,具有一定參考價值,需要的朋友可以了解下。
    2017-09-09

最新評論