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。
推薦閱讀:
- Java精確計算BigDecimal類詳解
- Java BigDecimal案例詳解
- BigDecimal的加減乘除計算方法詳解
- java BigDecimal類案例詳解
- java開發使用BigDecimal避坑四則