Java-BigDecimal數據類型

  • 2019 年 10 月 6 日
  • 筆記

我們知道在Java中有float和double類型,它們的主要設計目標是為了科學計算和工程計算。然而,它們沒有提供完全精確的結果【因為其有限的有效位數】,所以不應該被用於要求精確結果的場合。但是,商業計算往往要求更加精確的結果,這時候BigDecimal的重要性就不言而喻了!

BigDecimal簡介

BigDecimal 由任意精度的整數非標度值和32位的整數組成。BigDecimal所創建的是對象,我們不能使用傳統的+、-、*、/等算術運算符直接對其對象進行數學運算,而必須調用其相對應的方法。方法中的參數也必須是BigDecimal的對象。

BigDecimal一共有4個構造方法:

BigDecimal(int) 創建一個具有參數所指定整數值的對象。

BigDecimal(double) 創建一個具有參數所指定雙精度值的對象。

BigDecimal(long) 創建一個具有參數所指定長整數值的對象。

BigDecimal(String) 創建一個具有參數所指定以字符串表示的數值的對象。

BigDecimal 的運算方式,不支持 + – * / 這類的運算,它有自己的運算方法:

BigDecimal add(BigDecimal augend) 加法運算。

BigDecimal subtract(BigDecimal subtrahend) 減法運算。

BigDecimal multiply(BigDecimal multiplicand) 乘法運算。

BigDecimal divide(BigDecimal divisor) 除法運算。

示例代碼

構造函數(主要測試參數類型為double和String的兩個常用構造函數)

BigDecimal aDouble =new BigDecimal(1.22);

System.out.println("構造函數接受了一個double類型的值,結果是: " + aDouble);

BigDecimal aString = new BigDecimal("1.22");

System.out.println("構造函數接受了一個String類型的值,結果是: " + aString);

你認為輸出結果會是什麼呢?輸出結果如下:

構造函數接受了一個double類型的值,結果是

1.2199999999999999733546474089962430298328399658203125

構造函數接受了一個String類型的值,結果是: 1.22

在JDK中關於BigDecimal的描述

1、參數類型為double的構造方法的結果有一定的不可預知性。有人可能認為在Java中寫入newBigDecimal(0.1)所創建的BigDecimal正好等於 0.1(非標度值 1,其標度為 1),但它實際上等於0.1000000000000000055511151231257827021181583404541015625。這是因為0.1無法準確地表示為 double(或者說對於該情況,不能表示為任何有限長度的二進制小數)。所以,傳入到構造方法的值不會正好等於 0.1(雖然表面上等於該值)。

2、另一方面,String構造方法是完全可預知的:寫入newBigDecimal("0.1")將創建一個 BigDecimal,它正好等於預期的0.1。因此,比較而言,通常建議優先使用String構造方法。

3、當double必須用作BigDecimal的源時,請注意,此構造方法提供了一個準確轉換;它不提供與以下操作相同的結果:先使用Double.toString(double)方法,然後使用

BigDecimal (String)構造方法,將double轉換為String。如果要獲取該結果,請使用static valueOf(double)方法。

加法操作

BigDecimal a = new BigDecimal("1.22");

System.out.println("構造函數接受了一個String類型的值,結果是: " + a);

BigDecimal b = new BigDecimal("2.22");

a.add(b);

System.out.println("此時a的值是 : " + a);

我們很容易會認為會輸出:

構造函數接受了一個String類型的值,結果是 : 1.22

此時a的值是 : 3.44

但實際上a調用方法後的值是 : 1.22

總結

(1)商業計算一般會使用BigDecimal,因為涉及到金額、數據,所以必須有超高精度要求。

(2)盡量使用參數類型為String的構造函數。

(3)BigDecimal都是不可變的,在進行每一步運算時,都會產生一個新的對象,所以在做加減乘除運算時千萬要保存操作後的值。

(4)我們往往容易忽略JDK底層的一些實現細節,導致出現錯誤,需要多加註意。

更多乾貨筆記關注微信公眾號 : 老九學堂