BigDecimal簡介
JDK文檔(中文)中的解釋如下:
不可變的、任意精度的有符號十進制數。 BigDecimal 由任意精度的整數非標度值和32 位的整數標度(scale) 組成。如果為零或正數,則標度是小數點後的位數。如果為負數,則將該數的非標度值乘以10 的負scale 次冪。因此,BigDecimal 表示的數值是(unscaledValue × 10-scale)。
具體解釋
1.“BigDecimal 對象的值是不可變的”。這點在BigDecimal 對象的運算函數中表現了該特性:
複製代碼代碼如下:BigDecimal a = new BigDecimal("1.22");
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);
我們很容易會認為會輸出:
construct with a Stringvalue: 1.22
a plus b is :3.44
但實際上a plus b is : 1.22
2.“BigDecimal 由任意精度的整數非標度值和32 位的整數標度(scale) 組成。如果標度值為零或正數,則標度是小數點後的位數”。這句話可以這樣看:
例如:-12 和13.412
表示為:-12 × 10-0 和13412 × 10-3
這裡用(非標度值和標度)表示分別為:[-12, 0]和[13412, 3]
3.“如果標度值為負數,則將該數的非標度值乘以10 的負scale 次冪”。這句話可以這樣看:
例如:120.00
該值表示為:12000 × 10-2
這裡用(非標度值和標度)表示分別為:[12000, 2]
這裡標度的值依然為正數2,但是進行進行下面操作:
BigDecimal amount = new BigDecimal("-120.00");
//返回數值上等於此小數,但從該表示形式移除所有尾部零的BigDecimal。
amount = amount.stripTrailingZeros();
該值表示為:12 × 10-(-1)
這裡用(非標度值和標度)表示分別為:[12, -1]
使用注意事項
1.構造函數
複製代碼代碼如下:BigDecimal aDouble =new BigDecimal(1.22);
System.out.println("construct with a double value: " + aDouble);
BigDecimal aString = new BigDecimal("1.22");
System.out.println("construct with a String value: " + aString);
輸出結果如下:
construct with a doublevalue:1.2199999999999999733546474089962430298328399658203125
construct with a String value: 1.22
JDK的描述:
a)參數類型為double的構造方法的結果有一定的不可預知性。有人可能認為在Java中寫入newBigDecimal(0.1)所創建的BigDecimal正好等於0.1(非標度值1,其標度為1),但是它實際上等於0.1000000000000000055511151231257827021181583404541015625。這是因為0.1無法準確地表示為double(或者說對於該情況,不能表示為任何有限長度的二進制小數)。這樣,傳入到構造方法的值不會正好等於0.1(雖然表面上等於該值)。
b)另一方面,String 構造方法是完全可預知的:寫入newBigDecimal("0.1") 將創建一個BigDecimal,它正好等於預期的0.1。因此,比較而言,通常建議優先使用String構造方法。
c)當double必須用作BigDecimal的源時,請注意,此構造方法提供了一個準確轉換;它不提供與以下操作相同的結果:先使用Double.toString(double)方法,然後使用BigDecimal(String)構造方法。將double轉換為String,也可以使用String的static方法:String.valueOf(double)。
2.運算操作。加減乘除其實最終都返回的是一個新的BigDecimal對象,因為BigDecimal都是不可變的(immutable)的,在進行每一步運算時,都會產生一個新的對象,所以a.add(b);雖然做了加法操作,但是a並沒有保存加操作後的值,正確的用法應該是a=a.add(b);
例子:
判定BigDecimal 對像是否為整數:
複製代碼代碼如下:private boolean isIntegerValue(BigDecimal bd) {
return bd.signum() == 0 || bd.scale() <= 0 || bd.stripTrailingZeros().scale() <= 0;
}
為什麼要這樣做,請測試下下面這個例子:
複製代碼代碼如下:BigDecimal amount = new BigDecimal("-120.00");//請分別嘗試"0", "0.00", "1.00","10.00"和"10.10"
System.out.println(amount.signum());//正負
System.out.println(amount.scale()); //標度
System.out.println(amount.stripTrailingZeros().scale());//去零後的標度
參考:類BigDecimal