BigDecimal簡介
JDK文檔(中文)中的解釋如下:
不可變的、任意精度的有符號十進制數。BigDecimal 由任意精度的整數非標度值 和 32 位的整數標度 (scale) 組成。如果為零或正數,則標度是小數點后的位數。如果為負數,則將該數的非標度值乘以 10 的負 scale 次冪。因此,BigDecimal 表示的數值是 (unscaledValue × 10-scale)。
具體解釋
1.“BigDecimal 對象的值是不可變的”。這點在BigDecimal 對象的運算函數中表現了該特性:
我們很容易會認為會輸出:
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.構造函數
輸出結果如下:
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 對象是否為整數:
為什么要這樣做,請測試下下面這個例子:
參考:類 BigDecimal
新聞熱點
疑難解答