BigDecimal簡(jiǎn)介
JDK文檔(中文)中的解釋如下:
不可變的、任意精度的有符號(hào)十進(jìn)制數(shù)。BigDecimal 由任意精度的整數(shù)非標(biāo)度值 和 32 位的整數(shù)標(biāo)度 (scale) 組成。如果為零或正數(shù),則標(biāo)度是小數(shù)點(diǎn)后的位數(shù)。如果為負(fù)數(shù),則將該數(shù)的非標(biāo)度值乘以 10 的負(fù) scale 次冪。因此,BigDecimal 表示的數(shù)值是 (unscaledValue × 10-scale)。
具體解釋
1.“BigDecimal 對(duì)象的值是不可變的”。這點(diǎn)在BigDecimal 對(duì)象的運(yùn)算函數(shù)中表現(xiàn)了該特性:
System.out.println("construct with a String value: " + a);
BigDecimal b = new BigDecimal("2.22");
a.add(b);
System.out.println("a plus b is : " + a);
我們很容易會(huì)認(rèn)為會(huì)輸出:
construct with a Stringvalue: 1.22
a plus b is :3.44
但實(shí)際上a plus b is : 1.22
2.“BigDecimal 由任意精度的整數(shù)非標(biāo)度值 和 32 位的整數(shù)標(biāo)度 (scale) 組成。如果標(biāo)度值為零或正數(shù),則標(biāo)度是小數(shù)點(diǎn)后的位數(shù)”。這句話可以這樣看:
例如:-12 和 13.412
表示為:-12 × 10-0 和13412 × 10-3
這里用(非標(biāo)度值 和 標(biāo)度)表示分別為:[-12, 0]和[13412, 3]
3.“如果標(biāo)度值為負(fù)數(shù),則將該數(shù)的非標(biāo)度值乘以 10 的負(fù) scale 次冪”。這句話可以這樣看:
例如:120.00
該值表示為:12000 × 10-2
這里用(非標(biāo)度值 和 標(biāo)度)表示分別為:[12000, 2]
這里標(biāo)度的值依然為正數(shù)2,但是進(jìn)行進(jìn)行下面操作:
BigDecimal amount = new BigDecimal("-120.00");
//返回?cái)?shù)值上等于此小數(shù),但從該表示形式移除所有尾部零的 BigDecimal。
amount = amount.stripTrailingZeros();
該值表示為:12 × 10-(-1)
這里用(非標(biāo)度值 和 標(biāo)度)表示分別為:[12, -1]
使用注意事項(xiàng)
1.構(gòu)造函數(shù)
System.out.println("construct with a double value: " + aDouble);
BigDecimal aString = new BigDecimal("1.22");
System.out.println("construct with a String value: " + aString);
輸出結(jié)果如下:
construct with a doublevalue:1.2199999999999999733546474089962430298328399658203125
construct with a String value: 1.22
JDK的描述:
a)參數(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無法準(zhǔn)確地表示為 double(或者說對(duì)于該情況,不能表示為任何有限長(zhǎng)度的二進(jìn)制小數(shù))。這樣,傳入到構(gòu)造方法的值不會(huì)正好等于 0.1(雖然表面上等于該值)。
b)另一方面,String 構(gòu)造方法是完全可預(yù)知的:寫入 newBigDecimal("0.1") 將創(chuàng)建一個(gè) BigDecimal,它正好等于預(yù)期的 0.1。因此,比較而言,通常建議優(yōu)先使用String構(gòu)造方法。
c)當(dāng)double必須用作BigDecimal的源時(shí),請(qǐng)注意,此構(gòu)造方法提供了一個(gè)準(zhǔn)確轉(zhuǎn)換;它不提供與以下操作相同的結(jié)果:先使用Double.toString(double)方法,然后使用BigDecimal(String)構(gòu)造方法。將double轉(zhuǎn)換為String,也可以使用String的static方法:String.valueOf(double)。
2.運(yùn)算操作。加減乘除其實(shí)最終都返回的是一個(gè)新的BigDecimal對(duì)象,因?yàn)锽igDecimal都是不可變的(immutable)的,在進(jìn)行每一步運(yùn)算時(shí),都會(huì)產(chǎn)生一個(gè)新的對(duì)象,所以a.add(b);雖然做了加法操作,但是a并沒有保存加操作后的值,正確的用法應(yīng)該是a=a.add(b);
例子:
判定BigDecimal 對(duì)象是否為整數(shù):
return bd.signum() == 0 || bd.scale() <= 0 || bd.stripTrailingZeros().scale() <= 0;
}
為什么要這樣做,請(qǐng)測(cè)試下下面這個(gè)例子:
System.out.println(amount.signum());//正負(fù)
System.out.println(amount.scale()); //標(biāo)度
System.out.println(amount.stripTrailingZeros().scale());//去零后的標(biāo)度