溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

在Java中將double轉換為BigDecimal時需要注意哪些事項

發(fā)布時間:2021-01-20 13:46:07 來源:億速云 閱讀:860 作者:Leah 欄目:開發(fā)技術

今天就跟大家聊聊有關在Java中將double轉換為BigDecimal時需要注意哪些事項,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

先上結論:

不要直接用double變量作為構造BigDecimal的參數(shù)。

線上有這么一段Java代碼邏輯:

1,接口傳來一個JSON串,里面有個數(shù)字:57.3。

2,解析JSON并把這個數(shù)字保存在一個float變量。

3,把這個float變量賦值給一個 BigDecimal對象,用的是BigDecimal的double參數(shù)的構造:

new BigDecimal(double val)

4,把這個BigDecimal保存到MySQL數(shù)據(jù)庫,字段類型是decimal(15,2)。

這段代碼邏輯在線上跑了好久了,數(shù)據(jù)庫保存的值是57.3也沒什么問題,但是在今天debug的時候發(fā)現(xiàn),第三步的BigDecimal對象保存的值并不是57.3,而是57.299999237060546875,很明顯,出現(xiàn)了精度的問題。

至于數(shù)據(jù)庫最終保存了正確的57.3完全是因為字段類型設置為2位小數(shù),超過2位小數(shù)就四舍五入,所以才得到了正確的結果,相當于MySQL給我們把這個精度問題掩蓋了。

總覺得這是個坑,所以研究了一下相關的知識。

首先是BigDecimal的double參數(shù)構造,在官方JDK文檔中對這個構造是這么描述的:

public BigDecimal(double val)

Translates a double into a BigDecimal which is the exact decimal representation of the double's binary floating-point value. The scale of the returned BigDecimal is the smallest value such that (10scale × val) is an integer.

Notes:

The results of this constructor can be somewhat unpredictable. One might assume that writing new BigDecimal(0.1) in Java creates a BigDecimal which is exactly equal to 0.1 (an unscaled value of 1, with a scale of 1), but it is actually equal to 0.1000000000000000055511151231257827021181583404541015625. This is because 0.1 cannot be represented exactly as a double (or, for that matter, as a binary fraction of any finite length). Thus, the value that is being passed in to the constructor is not exactly equal to 0.1, appearances notwithstanding.

The String constructor, on the other hand, is perfectly predictable: writing new BigDecimal("0.1") creates a BigDecimal which is exactly equal to 0.1, as one would expect. Therefore, it is generally recommended that the String constructor be used in preference to this one.

When a double must be used as a source for a BigDecimal, note that this constructor provides an exact conversion; it does not give the same result as converting the double to a String using the Double.toString(double) method and then using the BigDecimal(String) constructor. To get that result, use the static valueOf(double) method.

Parameters:

val - double value to be converted to BigDecimal.

Throws:

NumberFormatException - if val is infinite or NaN.

翻譯一下大概是這樣的:

1,BigDecimal(double val)構造,用double當參數(shù)來構造一個BigDecimal對象。

2,但是這個構造不太靠譜(unpredictable),你可能以為BigDecimal(0.1)就是妥妥的等于0.1,但是你以為你以為的就是你以為的?還真不是,BigDecimal(0.1)這貨實際上等于0.1000000000000000055511151231257827021181583404541015625,因為準確的來說0.1本身不能算是一個double(其實0.1不能代表任何一個定長二進制分數(shù))。

3,BigDecimal(String val)構造是靠譜的,BigDecimal(“0.1”)就是妥妥的等于0.1,推薦大家用這個構造。

4,如果你非得用一個double變量來構造一個BigDecimal,沒問題,我們貼心的提供了靜態(tài)方法valueOf(double),這個方法跟new Decimal(Double.toString(double))效果是一樣的。

說白了就是別直接拿double變量做參數(shù),最好使用String類型做參數(shù)或者使用靜態(tài)方法valueOf(double),我寫了個例子試了一下:

public static void main(String[] args) { 
   float a=57.3f; 
   BigDecimal decimalA=new BigDecimal(a); 
   System.out.println(decimalA);   
 
   double b=57.3; 
   BigDecimal decimalB=new BigDecimal(b); 
   System.out.println(decimalB);   
 
   double c=57.3; 
   BigDecimal decimalC=new BigDecimal(Double.toString(c)); 
   System.out.println(decimalC);   
 
   double d=57.3; 
   BigDecimal decimalD=BigDecimal.valueOf(d); 
   System.out.println(decimalD); 
  }

輸出結果:

57.299999237060546875
57.2999999999999971578290569595992565155029296875
57.3
57.3

以后還是盡量按照官方推薦的套路來,否則不知道什么時候又給自己挖坑了。

補充:double轉bigDecimal精度問題

float的精度 : 2^23 7位

double的精度: 2^52 16位

十進制 轉 二進制 存在精度差

double g= 12.35;
BigDecimal bigG=new BigDecimal(g).setScale(1, BigDecimal.ROUND_HALF_UP); //期望得到12.4
System.out.println(“test G:”+bigG.doubleValue());
test G:12.3

原因:

定義double g= 12.35; 而在計算機中二進制表示可能這是樣:定義了一個g=12.34444444444444449,

new BigDecimal(g) g還是12.34444444444444449
new BigDecimal(g).setScale(1, BigDecimal.ROUND_HALF_UP); 得到12.3

正確的定義方式是使用字符串構造函數(shù):

new BigDecimal(“12.35”).setScale(1, BigDecimal.ROUND_HALF_UP)

首先得從計算機本身去討論這個問題。我們知道,計算機并不能識別除了二進制數(shù)據(jù)以外的任何數(shù)據(jù)。無論我們使用何種編程語言,在何種編譯環(huán)境下工作,都要先 把源程序翻譯成二進制的機器碼后才能被計算機識別。以上面提到的情況為例,我們源程序里的2.4是十進制的,計算機不能直接識別,要先編譯成二進制。但問 題來了,2.4的二進制表示并非是精確的2.4,反而最為接近的二進制表示是2.3999999999999999。原因在于浮點數(shù)由兩部分組成:指數(shù)和尾數(shù),這點如果知道怎樣進行浮點數(shù)的二進制與十進制轉換,應該是不難理解的。如果在這個轉換的過程中,浮點數(shù)參與了計算,那么轉換的過程就會變得不可預 知,并且變得不可逆。我們有理由相信,就是在這個過程中,發(fā)生了精度的丟失。而至于為什么有些浮點計算會得到準確的結果,應該也是碰巧那個計算的二進制與 十進制之間能夠準確轉換。而當輸出單個浮點型數(shù)據(jù)的時候,可以正確輸出,如

double d = 2.4;
System.out.println(d);

輸出的是2.4,而不是2.3999999999999999。也就是說,不進行浮點計算的時候,在十進制里浮點數(shù)能正確顯示。這更印證了我以上的想法,即如果浮點數(shù)參與了計算,那么浮點數(shù)二進制與十進制間的轉換過程就會變得不可預知,并且變得不可逆。

事實上,浮點數(shù)并不適合用于精確計算,而適合進行科學計算。這里有一個小知識:既然float和double型用來表示帶有小數(shù)點的數(shù),那為什么我們不稱 它們?yōu)椤靶?shù)”或者“實數(shù)”,要叫浮點數(shù)呢?因為這些數(shù)都以科學計數(shù)法的形式存儲。當一個數(shù)如50.534,轉換成科學計數(shù)法的形式為5.053e1,它 的小數(shù)點移動到了一個新的位置(即浮動了)??梢姡↑c數(shù)本來就是用于科學計算的,用來進行精確計算實在太不合適了。

在《Effective Java》這本書中也提到這個原則,float和double只能用來做科學計算或者是工程計算,在商業(yè)計算中我們要用java.math.BigDecimal。使用BigDecimal并且一定要用String來夠造。

BigDecimal用哪個構造函數(shù)?

BigDecimal(double val)
BigDecimal(String val)

看完上述內(nèi)容,你們對在Java中將double轉換為BigDecimal時需要注意哪些事項有進一步的了解嗎?如果還想了解更多知識或者相關內(nèi)容,請關注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

AI