java中BigDecimal用法詳解

首先,學習一個東西,我們都必須要帶著問題去學,這邊我分為 【為什麼?】【是什麼?】【怎麼用?】

【為什麼要用BigDecimal?】

首先,我們先看一下,下面這個現象

那為什麼會出現這種情況呢?

因為不論是float 還是double都是浮點數,而計算機是二進制的,浮點數會失去一定的精確度。

註:根本原因是:十進制值通常沒有完全相同的二進制表示形式;十進制數的二進制表示形式可能不精確。隻能無限接近於那個值

但是,在項目中,我們不可能讓這種情況出現,特別是金融項目,因為涉及金額的計算都必須十分精確,你想想,如果你的支付寶賬戶餘額顯示193.99999999999998,那是一種怎麼樣的體驗?

【BigDecimal是什麼?】

1、簡介

Java在java.math包中提供的API類BigDecimal,用來對超過16位有效位的數進行精確的運算。雙精度浮點型變量double可以處理16位有效數。在實際應用中,需要對更大或者更小的數進行運算和處理。float和double隻能用來做科學計算或者是工程計算,在商業計算中要用java.math.BigDecimal。BigDecimal所創建的是對象,我們不能使用傳統的+、-、*、/等算術運算符直接對其對象進行數學運算,而必須調用其相對應的方法。方法中的參數也必須是BigDecimal的對象。構造器是類的特殊方法,專門用來創建對象,特別是帶有參數的對象。

2、構造器描述 

BigDecimal(int)       創建一個具有參數所指定整數值的對象。 

BigDecimal(double) 創建一個具有參數所指定雙精度值的對象。 //不推薦使用

BigDecimal(long)    創建一個具有參數所指定長整數值的對象。 

BigDecimal(String) 創建一個具有參數所指定以字符串表示的數值的對象。//推薦使用

3、方法描述 

add(BigDecimal)        BigDecimal對象中的值相加,然後返回這個對象。 

subtract(BigDecimal) BigDecimal對象中的值相減,然後返回這個對象。 

multiply(BigDecimal)  BigDecimal對象中的值相乘,然後返回這個對象。 

divide(BigDecimal)     BigDecimal對象中的值相除,然後返回這個對象。 

toString()                將BigDecimal對象的數值轉換成字符串。 

doubleValue()          將BigDecimal對象中的值以雙精度數返回。 

floatValue()             將BigDecimal對象中的值以單精度數返回。 

longValue()             將BigDecimal對象中的值以長整數返回。 

intValue()               將BigDecimal對象中的值以整數返回。

特別說明一下,為什麼BigDecimal(double)  不推薦使用

看上面代碼運行結果,你就應該知道為什麼不推薦使用瞭,因為用這種方式也會導致計算有問題,

為什麼會出現這種情況呢?

 JDK的描述:

1、參數類型為double的構造方法的結果有一定的不可預知性。有人可能認為在Java中寫入newBigDecimal(0.1)所創建的BigDecimal正好等於 0.1(非標度值 1,其標度為 1),但是它實際上等於0.1000000000000000055511151231257827021181583404541015625。這是因為0.1無法準確地表示為 double(或者說對於該情況,不能表示為任何有限長度的二進制小數)。這樣,傳入到構造方法的值不會正好等於 0.1(雖然表面上等於該值)。

2、另一方面,String 構造方法是完全可預知的:寫入 newBigDecimal(“0.1”) 將創建一個 BigDecimal,它正好等於預期的 0.1。因此,比較而言,通常建議優先使用String構造方法

double必須用作BigDecimal的源時,請使用Double.toString(double)轉成String,然後使用String構造方法,或使用BigDecimal的靜態方法valueOf,如下

【怎麼用?】

這邊我就不多說什麼瞭,直接上代碼,都挺簡單的,最基本的加減乘除,應該能看的懂

這邊特別提一下,如果進行除法運算的時候,結果不能整除,有餘數,這個時候會報java.lang.ArithmeticException: ,這邊我們要避免這個錯誤產生,在進行除法運算的時候,針對可能出現的小數產生的計算,必須要多傳兩個參數divide(BigDecimal,保留小數點後幾位小數舍入模式)

舍入模式

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進行截斷和四舍五入可用setScale方法,例:

參考:

https://www.jb51.net/article/231683.htm

//www.jb51.net/article/231687.htm

以上就是本文的全部內容,希望對大傢的學習有所幫助,也希望大傢多多支持WalkonNet。

推薦閱讀: