1. 程式人生 > 實用技巧 >java 中 BigDecimal 詳解

java 中 BigDecimal 詳解

https://blog.csdn.net/qq_35868412/article/details/89029288

【為什麼要用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.cnblogs.com/LeoBoy/p/6056394.html

https://www.cnblogs.com/linjiqin/p/3413894.html