Java中關於 BigDecimal 的一個導致double精度損失的"bug"

来源:http://www.cnblogs.com/shouce/archive/2016/03/14/5274478.html
-Advertisement-
Play Games

背景      在博客 噁心的0.5四捨五入問題 一文中看到一個關於 0.5 不能正確的四捨五入的問題。主要說的是 double 轉換到 BigDecimal 後,進行四捨五入得不到正確的結果: 輸出的結果為: 301353.0499999999883584678173065185546875301


背景

     在博客 噁心的0.5四捨五入問題 一文中看到一個關於 0.5 不能正確的四捨五入的問題。主要說的是 double 轉換到 BigDecimal 後,進行四捨五入得不到正確的結果:

複製代碼
public class BigDecimalTest {
    public static void main(String[] args){
        double d = 301353.05;
        BigDecimal decimal = new BigDecimal(d);
        System.out.println(decimal);//301353.0499999999883584678173065185546875
        System.out.println(decimal.setScale(1, RoundingMode.HALF_UP));//301353.0
    }
}
複製代碼

輸出的結果為:

301353.0499999999883584678173065185546875
301353.0

這個結果顯然不是我們所期望的,我們希望的是得到 301353.1 。

原因

     允許明眼人一眼就看出另外問題所在——BigDecimal的構造函數 public BigDecimal(double val) 損失了double 參數的精度,最後才導致了錯誤的結果。所以問題的關鍵是:BigDecimal的構造函數 public BigDecimal(double val) 損失了double 參數的精度。

解決之道

因為上面找到了原因,所以也就很好解決了。只要防止了 double 到 BigDecimal 的精度的損失,也就不會出現問題。

1)很容易想到第一個解決辦法:使用BigDecimal的以String為參數的構造函數:public BigDecimal(String val)  來替代。

複製代碼
public class BigDecimalTest {
    public static void main(String[] args){
        double d = 301353.05;
        System.out.println(new BigDecimal(new Double(d).toString()));
        System.out.println(new BigDecimal("301353.05"));
        System.out.println(new BigDecimal("301353.895898895455898954895989"));
    }
}
複製代碼

輸出結果:

301353.05
301353.05
301353.895898895455898954895989

我們看到了沒有任何的精度損失,四捨五入也就肯定不會出錯了。

2)BigDecimal的構造函數 public BigDecimal(double val) 會損失了double 參數的精度,這個也許應該可以算作是 JDK 中的一個 bug 了。既然存在bug,那麼我們就應該解決它。上面的辦法是繞過了它。現在我們實現自己的 double 到 BigDecimal 的轉換,並且保證在某些情況下可以完全不損失 double 的精度。

複製代碼
import java.math.BigDecimal;

public class BigDecimalUtil {
    
    public static BigDecimal doubleToBigDecimal(double d){
        String doubleStr = String.valueOf(d);
        if(doubleStr.indexOf(".") != -1){
            int pointLen = doubleStr.replaceAll("\\d+\\.", "").length();    // 取得小數點後的數字的位數
            pointLen = pointLen > 16 ? 16 : pointLen;    // double最大有效小數點後的位數為16
            double pow = Math.pow(10, pointLen);
       long tmp = (long)(d * pow); return new BigDecimal(tmp).divide(new BigDecimal(pow)); } return new BigDecimal(d); } public static void main(String[] args){ // System.out.println(doubleToBigDecimal(301353.05)); // System.out.println(doubleToBigDecimal(-301353.05)); // System.out.println(doubleToBigDecimal(new Double(-301353.05))); // System.out.println(doubleToBigDecimal(301353)); // System.out.println(doubleToBigDecimal(new Double(-301353))); double d = 301353.05;//5898895455898954895989; System.out.println(doubleToBigDecimal(d)); System.out.println(d); System.out.println(new Double(d).toString()); System.out.println(new BigDecimal(new Double(d).toString())); System.out.println(new BigDecimal(d)); } }
複製代碼

輸出結果:

301353.05
301353.05
301353.05
301353.05
301353.0499999999883584678173065185546875

上面我們自己寫了一個工具類,實現了 double 到 BigDecimal 的“無損失”double精度的轉換。方法是將小數點後有有效數字的double先轉換到小數點後沒有有效數字的double,然後在轉換到 BigDecimal ,之後使用BigDecimal的 divide 返回之前的大小。

上面的結果看起來好像十分的完美,但是其實是存在問題的。上面我們也說到了“某些情況下可以完全不損失 double 的精度”,我們先看一個例子:

複製代碼
    public static void main(String[] args){
        double d = 301353.05;
        System.out.println(doubleToBigDecimal(d));
        System.out.println(d);
        System.out.println(new Double(d).toString());
        System.out.println(new BigDecimal(new Double(d).toString()));
        System.out.println(new BigDecimal(d));
        
        System.out.println("=========================");
        d = 301353.895898895455898954895989;
        System.out.println(doubleToBigDecimal(d));
        System.out.println(d);
        System.out.println(new Double(d).toString());
        System.out.println(new BigDecimal(new Double(d).toString()));
        System.out.println(new BigDecimal(d));
        System.out.println(new BigDecimal("301353.895898895455898954895989"));
        
        System.out.println("=========================");
        d = 301353.46899434;
        System.out.println(doubleToBigDecimal(d));
        System.out.println(d);
        System.out.println(new Double(d).toString());
        System.out.println(new BigDecimal(new Double(d).toString()));
        System.out.println(new BigDecimal(d));
        
        System.out.println("=========================");
        d = 301353.45789666;
        System.out.println(doubleToBigDecimal(d));
        System.out.println(d);
        System.out.println(new Double(d).toString());
        System.out.println(new BigDecimal(new Double(d).toString()));
        System.out.println(new BigDecimal(d));
    }
複製代碼

輸出結果:

301353.05
301353.05
301353.05
301353.05
301353.0499999999883584678173065185546875
=========================
301353.89589889544
301353.89589889545
301353.89589889545
301353.89589889545
301353.895898895454593002796173095703125
301353.895898895455898954895989
=========================
301353.46899434
301353.46899434
301353.46899434
301353.46899434
301353.4689943399862386286258697509765625
=========================
301353.45789666
301353.45789666
301353.45789666
301353.45789666
301353.4578966600238345563411712646484375
我們可以看到:我們自己實現的 doubleToBigDecimal 方法只有在 double 的小數點後的數字位數比較少時(比如只有5,6位),才能保證完全的不損失精度

在 double 的小數點後的數字位數比較多時,d * pow 會存在精度損失,所以最終的結果也會存在精度損失。所以如果小數點後的位數比較多時,還是使用 BigDecimal的 String 參數的構造函數為好,只有在小數點後的位數比較少時,才可以採用自己實現的 doubleToBigDecimal 方法。

因為我們看到原始的double的轉換之後的BigDecimal的數字的最後一位一個時5,一個是4,原因是在上面的轉換方法中:

long tmp = (long)(d * pow);

這一步可能存在很小的精度損失,因為 d 是一個 double ,d * pow 之後還是一個 double(但是小數點之後都是0了,所以到long的轉換沒有精度損失) ,所以會存在很小的精度損失(double的計算總是有可能存在精度損失的)。但是這個精度損失和 BigDecimal的構造函數 public BigDecimal(double val) 的精度損失相比而言,不會顯得那麼的突兀(也許我們自己寫的doubleToBigDecimal也是存在問題的,歡迎指點)。

總結

如果需要保證精度,最好是不要使用BigDecimal的double參數的構造函數,因為存在損失double參數精度的可能,最好是使用BigDecimal的String參數的構造函數。最好是杜絕使用BigDecimal的double參數的構造函數。

 

後記:

     其實說這是BigDecimal的一個bug,有標題黨的嫌疑,最多可以算作是BigDecimal的一個“坑”。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 角色是網站中都有的一個功能,用來區分用戶的類型、劃分用戶的許可權,這次實現角色列表瀏覽、角色添加、角色修改和角色刪除。 目錄 奔跑吧,代碼小哥! MVC5網站開發之一 總體概述 MVC5 網站開發之二 創建項目 MVC5 網站開發之三 數據存儲層功能實現 MVC5 網站開發之四 業務邏輯層的架構和基本...
  • WebApi2上進行依賴註入,在百度里能搜到的的完整解決方案的文章少之又少,缺胳膊斷腿。 和MVC5依賴註入的不同之處,並且需要註意的地方,標記在註釋當中。上Global代碼: 也沒有太多需要解釋的地方,Controller中還是構造器註入。開發中已經親測有效。    可以收藏,以後查看。  
  • 自從上次分享《Redis到底該如何利用?》已經有1年多了,這1年經歷了不少。從碼了我們網站的第一行開始到現在,我們的緩存模塊也不斷在升級,這之中確實略有心得,最近也有朋友探討緩存,覺得可以總結下分享下拙見,期待能有更深入的研究。 緩存是什麼? 我時常在群里或者在社區里看到有人對緩存有諸多疑問,搞不清
  •   在函數式編程中,可以把函數看作數據。函數也可以作為參數,函數還可以返回函數。比如,LINQ就是基於函數式編程的。 語句式編程可能這樣寫:   而使用函數式表達式,可以簡化為:   再來看一個過濾和排序的例子:   函數式編程可以寫成如下:   或   可見,在LINQ中,一個表達式(函數)的返回
  • 以下是 .NET Framework 4.5 中 ADO.NET 的新增功能。 以下是 .NET Framework 4.5 中用於 SQL Server 的 .NET Framework 數據提供程式的新增功能: ConnectRetryCount 和 ConnectRetryInterval 連
  • PHP,是英文超級文本預處理語言Hypertext Preprocessor的縮寫。PHP 是一種 HTML 內嵌式的語言,是一種在伺服器端執行的嵌入HTML文檔的腳本語言,語言的風格有類似於C語言,被廣泛的運用。自從1994年PHP語言的創建,神奇般的被追捧為網站設計的首選語言。2000年PHP4
  • 在C#中沒有獨立的函數存在,只有類的(動態或靜態)方法這一概念,它指的是類中用於執行計算或其它行為的成員。在Python中,你可以使用類似C#的方式定義類的動態或靜態成員方法,因為它與C#一樣支持完全的面向對象編程。你也可以用過程式編程的方式來編寫Python程式,這時Python中的函數與類可以沒
  • 鄭重聲明:本文是筆者網上翻譯原文,部分有做添加說明,所有權歸原文作者! 地址:http://www.cprogramming.com/c++11/rvalue-references-and-move-semantics-in-c++11.html C++一直致力於生成快速的程式。不幸的是,直到C++
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...