日韩成人免费在线_国产成人一二_精品国产免费人成电影在线观..._日本一区二区三区久久久久久久久不

當前位置:首頁 > 科技  > 軟件

不掌握 BigDecimal 的四大坑你敢用嗎?

來源: 責編: 時間:2024-05-30 17:18:01 175觀看
導讀BigDecimal 是 Java 中的一個類,這個相信大家都是知道的。它的作用就是可以表示任意精度的十進制數,BigDecimal 提供了精確的數字運算,適用于需要高精度計算的場景,例如金融、貨幣或者稅收等涉及到金錢的地方。與 double

BigDecimal 是 Java 中的一個類,這個相信大家都是知道的。它的作用就是可以表示任意精度的十進制數,BigDecimal 提供了精確的數字運算,適用于需要高精度計算的場景,例如金融、貨幣或者稅收等涉及到金錢的地方。e4K28資訊網——每日最新資訊28at.com

與 double 和 float 不同的是,BigDecimal 對象在計算的過程中不會丟失精度,那么下面我們就來看下第一個坑,浮點精度的坑。e4K28資訊網——每日最新資訊28at.com

e4K28資訊網——每日最新資訊28at.com

一、浮點精度的坑

我們先來看一個例子:e4K28資訊網——每日最新資訊28at.com

    public static void main(String[] args) {        BigDecimal num1 = new BigDecimal("0.1");        BigDecimal num2 = new BigDecimal("0.10");        // false        System.out.println(num1.equals(num2));        // 0        System.out.println(num1.compareTo(num2));    }

compareTo 方法比較中,a.compareTo(b)e4K28資訊網——每日最新資訊28at.com

返回:e4K28資訊網——每日最新資訊28at.com

  • -1: a小于b
  • 0: a等于b
  • 1: a大于b。

在上方的代碼中,我們使用 new BigDecimal 的形式 new 了兩個 BigDecimal 對象,分別是 0.1 和0.10。e4K28資訊網——每日最新資訊28at.com

我們分別使用了 equals 與 compareTo 進行比較,當使用 equals 進行比較時,返回了 false,這是因為 equals 不僅比較了值是否相等,還比較了精度是否相等,源碼中是這樣寫的:e4K28資訊網——每日最新資訊28at.com

 public boolean equals(Object x) {        if (!(x instanceof BigDecimal))            return false;        BigDecimal xDec = (BigDecimal) x;        if (x == this)            return true;        if (scale != xDec.scale)            return false;        long s = this.intCompact;        long xs = xDec.intCompact;        if (s != INFLATED) {            if (xs == INFLATED)                xs = compactValFor(xDec.intVal);            return xs == s;        } else if (xs != INFLATED)            return xs == compactValFor(this.intVal);        return this.inflated().equals(xDec.inflated());    }

所以在使用 equals 進行比較兩個 BigDecimal 的大小時,一定要注意這一點了。e4K28資訊網——每日最新資訊28at.com

簡單概括一下,如果比較兩個 BigDecimal 對象的大小,那就使用 compareTo 方法;如果嚴格比較精度的大小,那就使用 equals 方法進行比較。e4K28資訊網——每日最新資訊28at.com

上面我們知道了如何比較兩個 BigDecimal 對象的大小,equals 比較的還有他們的精度,那么精度又是如何設置的呢,這塊有沒有坑呢?e4K28資訊網——每日最新資訊28at.com

二、設置精度的坑

有的同學可能會說了,設置精度還有啥坑啊,設置了精度就好了嗎,哎對,就是這個意思,在做 BigDecimal 對象計算的時候,一定要設置精度。相反,有的同學就不喜歡設置精度,那么這 BUG 不就來了嗎。e4K28資訊網——每日最新資訊28at.com

來看一個例子:e4K28資訊網——每日最新資訊28at.com

    public static void main(String[] args) {        BigDecimal num1 = new BigDecimal("1");        BigDecimal num2 = new BigDecimal("3");        BigDecimal result = num1.divide(num2); // 默認舍入模式為 UNNECESSARY,會拋出 ArithmeticException    }

上述的代碼在執行結束之后會報錯 ArithmeticException ,這是因為默認舍入模式為 UNNECESSARY,所以會拋出 ArithmeticException。e4K28資訊網——每日最新資訊28at.com

Exception in thread "main" java.lang.ArithmeticException: Non-terminating decimal expansion; no exact representable decimal result.

要解決這個異常也很容易,只需要加上精度即可。e4K28資訊網——每日最新資訊28at.com

    public static void main(String[] args) {        BigDecimal num1 = new BigDecimal("1");        BigDecimal num2 = new BigDecimal("3");        BigDecimal result = num1.divide(num2, 2,RoundingMode.HALF_UP);        // 輸出:0.33        System.out.println(result);    }

那么出現這個異常的原因是什么你考慮過嗎?為什么加了精度就不報錯了呢?e4K28資訊網——每日最新資訊28at.com

這個異常在源碼中也有說明:e4K28資訊網——每日最新資訊28at.com

e4K28資訊網——每日最新資訊28at.com

大概意思就是如果在做 divide 運算時,如果商是一個無限小數,而操作的結果是一個精確的數字,那么就會拋出該異常。e4K28資訊網——每日最新資訊28at.com

e4K28資訊網——每日最新資訊28at.com

不知道大家注意到一點沒有,就是上面做除法運算的時候,也就是 BigDecimal result = num1.divide(num2, 2,RoundingMode.HALF_UP); 這行代碼的位置,使用了一個新的變量 result 來接收結果值,因為 BigDecimal 是不可變的,因此每次進行運算都會創建一個新的 BigDecimal 對象,所以這一點也是需要注意的,創建的多了可能會產生大量的垃圾對象。e4K28資訊網——每日最新資訊28at.com

講完了精度與運算,那么你初始化的方式對嗎?e4K28資訊網——每日最新資訊28at.com

三、初始化的坑

先來看代碼:e4K28資訊網——每日最新資訊28at.com

BigDecimal num = new BigDecimal(0.1); // 使用雙精度浮點數構造System.out.println(num); // 輸出: 0.1000000000000000055511151231257827021181583404541015625BigDecimal num2 = new BigDecimal("0.1"); // 使用字符串構造System.out.println(num2); // 輸出: 0.1

在使用 new BigDecimal 構造器進行初始化的時候,如果有初始值,最好使用字符串的構造方法進行初始化。e4K28資訊網——每日最新資訊28at.com

e4K28資訊網——每日最新資訊28at.com

在使用 double 的構造器進行新建時,本身傳入的 0.1 就是浮點類型了,為了不丟失精度,在使用 new BigDecimal 新建時就把這個近似值完整的保留下來了。e4K28資訊網——每日最新資訊28at.com

或者就是 另外一種初始化方式 BigDecimal.valueOf(0.1);,通過看源碼可以發現,在 valueOf 的內部,將 Double 類型直接轉為了字符串了,因此也就不會存在精度丟失的問題了。e4K28資訊網——每日最新資訊28at.com

對于使用 new BigDecimal(0.1) 構造時,源碼中也已經說明了這個問題。e4K28資訊網——每日最新資訊28at.com

e4K28資訊網——每日最新資訊28at.com

大體意思就是生成的 BigDecimal 對象不是我們想要的 0.1,推薦使用 String 類型的構造方法。e4K28資訊網——每日最新資訊28at.com

e4K28資訊網——每日最新資訊28at.com

上面我們已經學會了如何初始化,如何運算,下一步就是如何用了,例如轉字符串,很多同學可能會說,轉字符串 toString() 不就好了,如果你也這樣想,那你單純了弟弟。e4K28資訊網——每日最新資訊28at.com

四、轉字符串的坑

還是先看一段代碼:e4K28資訊網——每日最新資訊28at.com

    public static void main(String[] args) {        BigDecimal a = BigDecimal.valueOf(89382389312389594.33822312317952678768725);        System.out.println(a.toString()); // 輸出:8.93823893123896E+16        String str = a.setScale(2, RoundingMode.HALF_UP).toString();        System.out.println(str); // 輸出: 89382389312389600.00    }

上面代碼中是一個非常大的數,我想把他轉為字符串,可是在使用 toString() 方法時,打印出來的卻是科學計數法。e4K28資訊網——每日最新資訊28at.com

所以如果想使用 toString() 方法進行轉字符串時,可以使用設置精度的方法,但是結果還是與我們的預期有所差別,我們想要的是一模一樣的打印出來呢?e4K28資訊網——每日最新資訊28at.com

那么 toPlainString 就上場了,這個方法返回一個字符串的表示形式,包含所有的有效數字。e4K28資訊網——每日最新資訊28at.com

代碼修改如下:e4K28資訊網——每日最新資訊28at.com

    public static void main(String[] args) {        BigDecimal a = BigDecimal.valueOf(89382389312389594.99933822312317952678768725);        System.out.println(a.toPlainString());    }

修改之后就可以了嗎,不可以,忘了上面說的嗎,使用 String 的構造函數吧兄弟,double 類型的構造函數會丟失精度的。e4K28資訊網——每日最新資訊28at.com

最終代碼如下:e4K28資訊網——每日最新資訊28at.com

    public static void main(String[] args) {        BigDecimal a = new BigDecimal("89382389312389594.99933822312317952678768725");        System.out.println(a.toPlainString());    }

除了上述兩種轉字符串的方法外,還有一種,就是 toEngineeringString,這個方法也是返回一個字符串,包含有效數字,但是它會使用工程計數法,科學計數法的一種變體,它使用數字的倍數來表示值,使得指數是 3 的倍數。例如,1000會顯示為"1E3",而不是"1E+3"。e4K28資訊網——每日最新資訊28at.com

所以總結就是:e4K28資訊網——每日最新資訊28at.com

  • toString:返回有效數字,必要的時候使用科學計數法。
  • toPlainString: 不實用任何科學計數法。
  • toEngineeringString:必要的時候使用工程計數法。

五、總結

本文從精度的比較、除法運算中是否設置精度、對象初始化到轉字符串,四個角度來把 BigDecimal 的坑盡可能清晰的描述出來,以及基于這些坑得到的優秀實踐。e4K28資訊網——每日最新資訊28at.com

有些場景下推薦使用 BigDecimal ,但是能不用還是不用,比 double 、float 多出來的性能損失得是你能接受的。如果非得用,那上面這幾個坑一定要規避。e4K28資訊網——每日最新資訊28at.com

本文鏈接:http://www.www897cc.com/showinfo-26-91825-0.html不掌握 BigDecimal 的四大坑你敢用嗎?

聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。郵件:2376512515@qq.com

上一篇: C語言字符串為什么以/0 作為結束標志?

下一篇: 從0到1,手把手教你部署自己的線上項目

標簽:
  • 熱門焦點
  • vivo TWS Air開箱體驗:真輕 臻好聽

    在vivo S15系列新機的發布會上,vivo的最新款真無線藍牙耳機vivo TWS Air也一同發布,本次就這款耳機新品給大家帶來一個簡單的分享。外包裝盒上,vivo TWS Air保持了vivo自家產
  • 6月iOS設備好評榜:第一蟬聯榜首近一年

    作為安兔兔各種榜單里變化最小的那個,2023年6月的iOS好評榜和上個月相比沒有任何排名上的變化,僅僅是部分設備好評率的下降,長年累月的用戶評價和逐漸退出市場的老款機器讓這
  • 8月總票房已突破10億!《封神》第一:口碑已經成了

    8月5日消息,據燈塔專業版數據,截至8月5日9時35分,8月總票房(含預售)已突破10億。其中,《封神》以大比分的優勢領先。根據官方消息,目前該片總票房已經超過14.
  • 如何通過Python線程池實現異步編程?

    線程池的概念和基本原理線程池是一種并發處理機制,它可以在程序啟動時創建一組線程,并將它們置于等待任務的狀態。當任務到達時,線程池中的某個線程會被喚醒并執行任務,執行完任
  • 每天一道面試題-CPU偽共享

    前言:了不起:又到了每天一到面試題的時候了!學弟,最近學習的怎么樣啊 了不起學弟:最近學習的還不錯,每天都在學習,每天都在進步! 了不起:那你最近學習的什么呢? 了不起學弟:最近在學習C
  • 最“俊美”淘寶賣家,靠直播和短視頻圈粉,上架秒光,年銷3000萬

    來源 | 電商在線文|易琬玉編輯|斯問受訪店鋪:Ringdoll戒之人形圖源:微博@御座的黃山、“Ringdoll戒之人形”淘寶店鋪有關外貌的評價,黃山已經聽累了。生于1985年的他,哪
  • 三星Galaxy Z Fold5今日亮相:厚度縮減但仍略顯厚重

    據官方此前宣布,三星將于7月26日也就是今天在韓國首爾舉辦Unpacked活動,屆時將帶來帶來包括Galaxy Buds 3、Galaxy Watch 6、Galaxy Tab S9、Galaxy
  • 蘋果MacBook Pro 2021測試:仍不支持平滑滾動

    據10月30日9to5 Mac 消息報道,蘋果新的 14 英寸和 16 英寸 MacBook Pro 2021 上市后獲得了不錯的評價,亮點包括行業領先的性能,令人印象深刻的電池續航,精美豐
  • 利用職權私自解除被封帳號 Meta開除20多名員工

    11月18日消息,據外媒援引知情人士表示,過去一年時間內,Facebook母公司Meta解雇或處罰了20多名員工以及合同工,指控這些人通過內部系統以不當方式重置用戶帳號,其
Top 主站蜘蛛池模板: 垣曲县| 大埔县| 绩溪县| 香港 | 无为县| 广灵县| 武汉市| 安福县| 青田县| 和龙市| 依安县| 云南省| 新乡县| 如东县| 洞头县| 民和| 昌邑市| 和硕县| 连云港市| 磴口县| 宜阳县| 民和| 龙胜| 巩义市| 南澳县| 罗城| 峨眉山市| 广平县| 松原市| 海伦市| 旌德县| 托克托县| 调兵山市| 内黄县| 武川县| 饶阳县| 额敏县| 盐亭县| 阿拉尔市| 宜川县| 眉山市|