用BigDecimal計算金額就高枕無憂了?帶你了解BigDecimal這五個坑
BigDecimal是java.math包中提供的API類,專門用于對超過16位有效位的數(shù)進(jìn)行精確的運(yùn)算。一般情況下,對于不需要準(zhǔn)確精度的計算,我們可以通過float或者double進(jìn)行計算,但是如果需要精確計算的結(jié)果,就必須使用BigDecimal類來操作。
而BigDecamal本質(zhì)上是一個對象,所以傳統(tǒng)的+、-、*、/等算術(shù)運(yùn)算符就不能直接使用了。
今天我們來了解一下,使用BigDecimal時容易踩的五個坑。
1、錯誤的初始化
BigDecimal提供了很多初始化方法,但是需要注意的是,當(dāng)我們使用double這種浮點(diǎn)型初始化,可能得到預(yù)期外的結(jié)果。
當(dāng)我們使用浮點(diǎn)型進(jìn)行初始化時,由于其本身就是一個不確定的值,傳給BigDecimal初始化時已經(jīng)丟失了精度,所以看起來就像一個bug。
一般來說,建議大家初始化時將其轉(zhuǎn)為String或者使用BigDecimal.valueOf方法。
2、錯誤的等值比較方法
equal方法會比較兩個BigDecimal對象的精度,如果精度不同,則認(rèn)為是兩個不同的對象。
如果要比較兩個對象的大小,建議使用compareTo方法。
3、未指定精度可能會導(dǎo)致異常
如果除法的商是一個無限小數(shù),而我們又沒有指定精度大小,程序?qū)伋鲆粋€異常。當(dāng)我們使用BigDecimal時,應(yīng)該時刻記得指定精度,避免因?yàn)榫葐栴}帶來的損失。
4、toString方法的格式問題
當(dāng)我們想把一個BigDecimal.valueOf構(gòu)造的BigDecimal轉(zhuǎn)為String時,應(yīng)該了解到toString方法和toPlainString方法的區(qū)別。
toString:必要時會使用科學(xué)計數(shù)法。
toPlainString:不使用科學(xué)計數(shù)法。
toEngineeringString:工程計算中經(jīng)常使用的記錄數(shù)字的方法,與科學(xué)計數(shù)法類似,但要求10的冪必須是3的倍數(shù)。
5、執(zhí)行順序不同導(dǎo)致的結(jié)果差異
當(dāng)執(zhí)行算術(shù)運(yùn)算時,會滿足乘法交換律,但是對于BigDecimal來說,是不適用的。
不同的執(zhí)行順序會得到不同的結(jié)果,對于金融類的需求,這0.1的差額,足夠你排除bug到半夜了。
總結(jié)
1、BigDecimal雖然計算精度準(zhǔn)確,但是其性能相對double,float是較差的。如果沒有高精度計算的要求,那也不必強(qiáng)行使用BigDecimal。
2、初始BigDecimal時,建議強(qiáng)制使用字符串的構(gòu)造參數(shù)。
3、BigDecimal對象是不可變的,每次計算都會產(chǎn)生一個新對象,所以記得保存做完計算以后的值。