Java編程BigDecimal用法實例分享
Java中提供了大數(shù)字(超過16位有效位)的操作類,即 java.math.BinInteger 類和 java.math.BigDecimal 類,用于高精度計算.
其中 BigInteger 類是針對大整數(shù)的處理類,而 BigDecimal 類則是針對大小數(shù)的處理類.
BigDecimal 類的實現(xiàn)用到了 BigInteger類,不同的是 BigDecimal 加入了小數(shù)的概念.
float和Double只能用來做科學計算或者是工程計算;在商業(yè)計算中,對數(shù)字精度要求較高,必須使用 BigInteger 類和 BigDecimal 類,它支持任何精度的定點數(shù),可以用它來精確計算貨幣值.
BigDecimal類創(chuàng)建的是對象,不能使用傳統(tǒng)的+、-、*、/等算術(shù)運算符直接對其進行數(shù)學運算,而必須調(diào)用其對應的方法.方法的參數(shù)也必須是BigDecimal類型的對象.
構(gòu)造 BigDecimal 對象常用以下方法:
BigDecimal BigDecimal(double d); //不允許使用
BigDecimal BigDecimal(String s); //常用,推薦使用
static BigDecimal valueOf(double d); //常用,推薦使用
其中,
1. double 參數(shù)的構(gòu)造方法,不允許使用!!!!因為它不能精確的得到相應的值;
2. String 構(gòu)造方法是完全可預知的: 寫入 new BigDecimal("0.1") 將創(chuàng)建一個 BigDecimal,它正好等于預期的0.1; 因此,通常建議優(yōu)先使用 String 構(gòu)造方法;
3. 靜態(tài)方法 valueOf(double val) 內(nèi)部實現(xiàn),仍是將 double 類型轉(zhuǎn)為 String 類型; 這通常是將 double(或float)轉(zhuǎn)化為 BigDecimal 的首選方法;
測試代碼如下:
public static void main(String[] args) { double d1 = 0.10334; double d2 = 1234.0; System.out.println("new BigDecimal("+d1+")=" + new BigDecimal(d1)); //此種方式絕對不允許!!!!! System.out.println("new BigDecimal("+d2+")=" + new BigDecimal(d2)); //此種方式絕對不允許!!!!! System.out.println(""); System.out.println("new BigDecimal(String.valueOf("+d1+"))=" + new BigDecimal(String.valueOf(d1))); System.out.println("new BigDecimal(String.valueOf("+d2+"))=" + new BigDecimal(String.valueOf(d2))); System.out.println(""); System.out.println("new BigDecimal(String.valueOf("+d1+"))=" + new BigDecimal(Double.toString(d1))); System.out.println("new BigDecimal(String.valueOf("+d2+"))=" + new BigDecimal(Double.toString(d2))); System.out.println(""); System.out.println("BigDecimal.valueOf("+d1+")=" + BigDecimal.valueOf(d1)); System.out.println("BigDecimal.valueOf("+d2+")=" + BigDecimal.valueOf(d2)); System.out.println(""); BigDecimal b1 = BigDecimal.valueOf(1); BigDecimal b2 = BigDecimal.valueOf(1.00000); System.out.println(b1.equals(b2)); System.out.println(b1.compareTo(b2)); }
輸出如下:
new BigDecimal(0.10334)=0.10334000000000000130118138486068346537649631500244140625 new BigDecimal(1234.0)=1234 new BigDecimal(String.valueOf(0.10334))=0.10334 new BigDecimal(String.valueOf(1234.0))=1234.0 new BigDecimal(String.valueOf(0.10334))=0.10334 new BigDecimal(String.valueOf(1234.0))=1234.0 BigDecimal.valueOf(0.10334)=0.10334 BigDecimal.valueOf(1234.0)=1234.0 false 0
附1, BigDecimal
類的 valueOf()
方法源碼
public static BigDecimal valueOf(double val) { return new BigDecimal(Double.toString(val)); }
附2, BigDecimal類的幾個常用方法
/** * 求余數(shù) * 返回值為 (this % divisor) 的 BigDecimal */ BigDecimal remainder(BigDecimal divisor); /** * 求相反數(shù) * 返回值是 (-this) 的 BigDecimal */ BigDecimal negate(); /** * 將此 BigDecimal 與指定的 BigDecimal 比較 * 根據(jù)此方法,值相等但具有不同標度的兩個 BigDecimal 對象(如,2.0 和 2.00)被認為是相等的; * 相對六個 boolean 比較運算符 (<, ==, >, >=, !=, <=) 中每一個運算符的各個方法,優(yōu)先提供此方法; * 建議使用以下語句執(zhí)行上述比較:(x.compareTo(y) <op> 0), 其中 <op> 是六個比較運算符之一; * * 指定者:接口 Comparable<BigDecimal> 中的 compareTo * 返回:當此 BigDecimal 在數(shù)字上小于、等于或大于 val 時,返回 -1、0 或 1 */ int compareTo(BigDecimal val);
附3, 提供精確的浮點數(shù)運算(包括加、減、乘、除、四舍五入)的工具類源碼
package com.util; import java.math.BigDecimal; /** * 提供精確的浮點數(shù)運算(包括加、減、乘、除、四舍五入)工具類 */ public class ArithUtil { // 除法運算默認精度 private static final int DEF_DIV_SCALE = 10; private ArithUtil() { } /** * 精確加法 */ public static double add(double value1, double value2) { BigDecimal b1 = BigDecimal.valueOf(value1); BigDecimal b2 = BigDecimal.valueOf(value2); return b1.add(b2).doubleValue(); } /** * 精確減法 */ public static double sub(double value1, double value2) { BigDecimal b1 = BigDecimal.valueOf(value1); BigDecimal b2 = BigDecimal.valueOf(value2); return b1.subtract(b2).doubleValue(); } /** * 精確乘法 */ public static double mul(double value1, double value2) { BigDecimal b1 = BigDecimal.valueOf(value1); BigDecimal b2 = BigDecimal.valueOf(value2); return b1.multiply(b2).doubleValue(); } /** * 精確除法 使用默認精度 */ public static double div(double value1, double value2) throws IllegalAccessException { return div(value1, value2, DEF_DIV_SCALE); } /** * 精確除法 * @param scale 精度 */ public static double div(double value1, double value2, int scale) throws IllegalAccessException { if(scale < 0) { throw new IllegalAccessException("精確度不能小于0"); } BigDecimal b1 = BigDecimal.valueOf(value1); BigDecimal b2 = BigDecimal.valueOf(value2); // return b1.divide(b2, scale).doubleValue(); return b1.divide(b2, scale, BigDecimal.ROUND_HALF_UP).doubleValue(); } /** * 四舍五入 * @param scale 小數(shù)點后保留幾位 */ public static double round(double v, int scale) throws IllegalAccessException { return div(v, 1, scale); } /** * 比較大小 */ public static boolean equalTo(BigDecimal b1, BigDecimal b2) { if(b1 == null || b2 == null) { return false; } return 0 == b1.compareTo(b2); } }
下面,總結(jié)下這次項目中BigDecimal的用法。
1.加減乘除
2.設(shè)置精度
3.取反
加減乘除分別調(diào)用函數(shù)
public BigDecimal add(BigDecimal value); public BigDecimal subtract(BigDecimal value); public BigDecimal multiply(BigDecimal value); public BigDecimal divide(BigDecimal value);
舉例:
BigDecimal a=new BigDecimal(10.0); BigDecimal b=new BigDecimal(9.1); System.out.println(a.subtract(b));
精度設(shè)置,為什么會設(shè)置精度,給大家看個效果
代碼如下
BigDecimal abig=new BigDecimal(10.0); BigDecimal bbig=new BigDecimal(9.1); BigDecimal cbig=new BigDecimal(8.9); System.out.println(abig.subtract(bbig)); System.out.println(abig.subtract(cbig));
結(jié)果如下:
0.9000000000000003552713678800500929355621337890625 1.0999999999999996447286321199499070644378662109375
并不是我們希望看到的0.9和1.1,原因就是轉(zhuǎn)成二進制的時候會有精度問題,導致這樣的結(jié)果。所以我們可以在運算的時候加精度,也可以在實例化BigDecimal的時候用字符串。
設(shè)置精度的方法:
System.out.println(abig.subtract(bbig).setScale(2, BigDecimal.ROUND_HALF_UP)); System.out.println(abig.subtract(cbig).setScale(2, BigDecimal.ROUND_HALF_UP));
這樣設(shè)置兩位精度就可以啦
0.90 1.10
字符串實例化的方法:
BigDecimal abig=new BigDecimal("10.0"); BigDecimal bbig=new BigDecimal("9.1"); BigDecimal cbig=new BigDecimal("8.9");
取反
因為Bigdecimal是無法直接用+-*/這些符號進行計算的,所以取反的時候也需要一個單獨的方法來實現(xiàn):
System.out.println(abig.negate());
這樣就會拿到它的相反數(shù)了:
-10.0
這些是這次項目中用到的點
總結(jié)
以上就是本文關(guān)于Java編程BigDecimal用法實例分享的全部內(nèi)容,希望對大家有所幫助。歡迎參閱:Java之dao模式詳解及代碼示例、java編程中自動拆箱與自動裝箱詳解、java數(shù)組基礎(chǔ)詳解等,有什么問題可以隨時留言,歡迎大家指出!
相關(guān)文章
詳解SpringMVC中的@RequestMapping注解
這篇文章主要介紹了SpringMVC中@RequestMapping注解,@RequestMapping注解是一個用來處理請求地址映射的注解,可用于映射一個請求或一個方法,可以用在類或方法上,需要的朋友可以參考下2023-07-07springboot如何獲取application.yml里值的方法
這篇文章主要介紹了springboot如何獲取application.yml里的值,文章圍繞主題相關(guān)自資料展開詳細的內(nèi)容介紹,具有一定的參考價值,需要的小伙伴可以參考一下2022-04-04SpringData JPA基本/高級/多數(shù)據(jù)源的使用詳解
這篇文章主要介紹了SpringData JPA基本/高級/多數(shù)據(jù)源的使用詳解,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-02-02SpringBoot調(diào)用service層的三種方法
在Spring?Boot中,我們可以通過注入Service層對象來調(diào)用Service層的方法,Service層是業(yè)務邏輯的處理層,它通常包含了對數(shù)據(jù)的增刪改查操作,本文給大家介紹了SpringBoot調(diào)用service層的三種方法,需要的朋友可以參考下2024-05-05解析Java的Spring框架的BeanPostProcessor發(fā)布處理器
這篇文章主要介紹了Java的Spring框架的BeanPostProcessor發(fā)布處理器,Spring是Java的SSH三大web開發(fā)框架之一,需要的朋友可以參考下2015-12-12Mybatis實現(xiàn)動態(tài)SQL編寫詳細代碼示例
這篇文章主要為大家詳細介紹了Mybatis中動態(tài)SQL的編寫使用,動態(tài)SQL技術(shù)是一種根據(jù)特定條件動態(tài)拼裝SQL語句的功能,它存在的意義是為了解決拼接SQL語句字符串時的痛點問題,感興趣想要詳細了解可以參考下文2023-05-05Springboot集成Actuator監(jiān)控功能詳解
這篇文章主要介紹了Springboot集成Actuator監(jiān)控功能詳解,有時候我們想要實時監(jiān)控我們的應用程序的運行狀態(tài),比如實時顯示一些指標數(shù)據(jù),觀察每時每刻訪問的流量,或者是我們數(shù)據(jù)庫的訪問狀態(tài)等等,這時候就需要Actuator了,需要的朋友可以參考下2023-09-09SpringMVC事件監(jiān)聽ApplicationListener實例解析
這篇文章主要介紹了SpringMVC事件監(jiān)聽ApplicationListener實例解析,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下2019-11-11