為什么要使用BigDecimal
說(shuō)到j(luò)ava中的浮點(diǎn)型,大家主要用的還是float和double,但這兩種浮點(diǎn)型的精度都是有限的,一般情況下只是一個(gè)近似值姆打,永遠(yuǎn)不要相信浮點(diǎn)數(shù)結(jié)果精確到了最后一位,也永遠(yuǎn)不要相信兩個(gè)浮點(diǎn)數(shù)是否相等肠虽。但在有些情況下幔戏,比如財(cái)務(wù)系統(tǒng)中,對(duì)于精度要求很高税课,這里Java提供了一個(gè)BigDecimal類闲延。
構(gòu)造方法
BigDecimal(int) 創(chuàng)建一個(gè)具有參數(shù)所指定整數(shù)值的對(duì)象。
BigDecimal(double) 創(chuàng)建一個(gè)具有參數(shù)所指定雙精度值的對(duì)象伯复。 //禁止使用
BigDecimal(long) 創(chuàng)建一個(gè)具有參數(shù)所指定長(zhǎng)整數(shù)值的對(duì)象慨代。
BigDecimal(String) 創(chuàng)建一個(gè)具有參數(shù)所指定以字符串表示的數(shù)值的對(duì)象。//推薦使用
為何禁止使用BigDecimal(double)
看上面代碼運(yùn)行結(jié)果啸如,你就應(yīng)該知道為什么不推薦使用了侍匙,因?yàn)橛眠@種方式也會(huì)導(dǎo)致計(jì)算有問題,
為什么會(huì)出現(xiàn)這種情況呢叮雳?
JDK的描述:1想暗、參數(shù)類型為double的構(gòu)造方法的結(jié)果有一定的不可預(yù)知性。有人可能認(rèn)為在Java中寫入newBigDecimal(0.1)所創(chuàng)建的BigDecimal正好等于 0.1(非標(biāo)度值 1帘不,其標(biāo)度為 1)说莫,但是它實(shí)際上等于0.1000000000000000055511151231257827021181583404541015625。這是因?yàn)?.1無(wú)法準(zhǔn)確地表示為 double(或者說(shuō)對(duì)于該情況寞焙,不能表示為任何有限長(zhǎng)度的二進(jìn)制小數(shù))储狭。這樣,傳入到構(gòu)造方法的值不會(huì)正好等于 0.1(雖然表面上等于該值)捣郊。
另一方面辽狈,String 構(gòu)造方法是完全可預(yù)知的:寫入 newBigDecimal("0.1") 將創(chuàng)建一個(gè) BigDecimal,它正好等于預(yù)期的 0.1呛牲。因此刮萌,比較而言,通常建議優(yōu)先使用String構(gòu)造方法
靜態(tài)方法 valueOf(double val) 內(nèi)部實(shí)現(xiàn),仍是將 double 類型轉(zhuǎn)為 String 類型; 這通常是將 double(或float)轉(zhuǎn)化為 BigDecimal 的首選方法
方法描述
add(BigDecimal) BigDecimal對(duì)象中的值相加娘扩,然后返回這個(gè)對(duì)象着茸。
subtract(BigDecimal) BigDecimal對(duì)象中的值相減壮锻,然后返回這個(gè)對(duì)象。
multiply(BigDecimal) BigDecimal對(duì)象中的值相乘涮阔,然后返回這個(gè)對(duì)象猜绣。
divide(BigDecimal) BigDecimal對(duì)象中的值相除,然后返回這個(gè)對(duì)象澎语。
toString() 將BigDecimal對(duì)象的數(shù)值轉(zhuǎn)換成字符串途事。
doubleValue() 將BigDecimal對(duì)象中的值以雙精度數(shù)返回。
floatValue() 將BigDecimal對(duì)象中的值以單精度數(shù)返回擅羞。
longValue() 將BigDecimal對(duì)象中的值以長(zhǎng)整數(shù)返回。
intValue() 將BigDecimal對(duì)象中的值以整數(shù)返回义图。
當(dāng)double必須用作BigDecimal的源時(shí)减俏,請(qǐng)使用Double.toString(double)轉(zhuǎn)成String,然后使用String構(gòu)造方法碱工,或使用BigDecimal的靜態(tài)方法valueOf娃承,如下
static BigDecimal valueOf(double d); //常用,推薦使用
使用方法
這邊特別提一下,如果進(jìn)行除法運(yùn)算的時(shí)候怕篷,結(jié)果不能整除历筝,有余數(shù),這個(gè)時(shí)候會(huì)報(bào)java.lang.ArithmeticException:廊谓,這邊我們要避免這個(gè)錯(cuò)誤產(chǎn)生梳猪,在進(jìn)行除法運(yùn)算的時(shí)候,針對(duì)可能出現(xiàn)的小數(shù)產(chǎn)生的計(jì)算蒸痹,必須要多傳兩個(gè)參數(shù)
divide(BigDecimal春弥,保留小數(shù)點(diǎn)后幾位小數(shù),舍入模式)
舍入模式
ROUND_CEILING //向正無(wú)窮方向舍入
ROUND_DOWN //向零方向舍入
ROUND_FLOOR //向負(fù)無(wú)窮方向舍入
ROUND_HALF_DOWN //向(距離)最近的一邊舍入叠荠,除非兩邊(的距離)是相等,如果是這樣匿沛,向下舍入, 例如1.55 保留一位小數(shù)結(jié)果為1.5
ROUND_HALF_EVEN //向(距離)最近的一邊舍入,除非兩邊(的距離)是相等,如果是這樣榛鼎,如果保留位數(shù)是奇數(shù)逃呼,使用ROUND_HALF_UP,如果是偶數(shù)者娱,使用ROUND_HALF_DOWN
ROUND_HALF_UP //向(距離)最近的一邊舍入抡笼,除非兩邊(的距離)是相等,如果是這樣,向上舍入, 1.55保留一位小數(shù)結(jié)果為1.6,也就是我們常說(shuō)的“四舍五入”
ROUND_UNNECESSARY //計(jì)算結(jié)果是精確的肺然,不需要舍入模式
ROUND_UP //向遠(yuǎn)離0的方向舍入
需要對(duì)BigDecimal進(jìn)行截?cái)嗪退纳嵛迦肟捎?strong>setScale方法蔫缸,例:
參考鏈接
java 中 BigDecimal 詳解