java float 加減精度問題


java float 加減精度問題
在取這個字段的時候轉換成BigDecimal就可以了
同時,BigDecimal是可以設置精度的。
float m = 12.22F;
float c = 1.22F;
BigDecimal b1 = new BigDecimal(Float.toString(m));
BigDecimal b2 = new BigDecimal(Float.toString(c));
System.out.println(m);
System.out.println(c);
Float add = b1.add(b2).floatValue();
System.out.println("add=========" + add);
Float less = b1.subtract(b2).floatValue();
System.out.println("less=========" + less);
Float multiply = b1.multiply(b2).floatValue();
System.out.println("multiply=========" + multiply);
//給divide設置精確的小數點,解決不整除報異常
Float divide = b1.divide(b2,4, BigDecimal.ROUND_HALF_EVEN).floatValue();
System.out.println("divide=========" + divide);
===========================================
BigDecimal不整除的一個異常java.lang.ArithmeticException: Non-terminating decimal expansion
金額的數據類型是BigDecimal 
通過BigDecimal的divide方法進行除法時當不整除,出現無限循環小數時,就會拋異常的,異常如下:java.lang.ArithmeticException:
 Non-terminating decimal expansion; no exact representable decimal result. at java.math.BigDecimal.divide(Unknown Source)
應用場景:一批中供客戶的單價是1000元/年,如果按月計算的話1000/12=83.3333333333.... 
解決之道:就是給divide設置精確的小數點divide(xxxxx,2, BigDecimal.ROUND_HALF_EVEN) 
在 java中, 四舍五入通過 BigDecimal 來實現。一定要注意:BigDecimal is Immutable。
也就是跟String一樣,對前一個的修改,比如setScale(), add()等都會返回一個新的BigDecimal.
四舍五入舍入模式是 BigDecimal.ROUND_HALF_UP 
BigDecimal定義了一下舍入模式,只有在作除法運算或四舍五入時才用到舍入模式,
下面簡單介紹,詳細請查閱J2se API文檔
static int
ROUND_CEILING
Rounding mode to round towards positive infinity.
向正無窮方向舍入
static int
ROUND_DOWN
Rounding mode to round towards zero.
向零方向舍入
static int
ROUND_FLOOR
Rounding mode to round towards negative infinity.
向負無窮方向舍入
static int
ROUND_HALF_DOWN
Rounding mode to round towards "nearest neighbor" unless both neighbors are equidistant,
in which case round down.
向(距離)最近的一邊舍入,除非兩邊(的距離)是相等,如果是這樣,向下舍入,例如1.55保留一位
小數結果為1.5
static int
ROUND_HALF_EVEN
Rounding mode to round towards the "nearest neighbor" unless both neighbors are equidistant,
in which case, round towards the even neighbor.
向(距離)最近的一邊舍入,除非兩邊(的距離)是相等,如果是這樣,如果保留位數是奇數,
使用ROUND_HALF_UP,如果是偶數,使用ROUND_HALF_DOWN
static int
ROUND_HALF_UP
Rounding mode to round towards "nearest neighbor" unless both neighbors are equidistant,
in which case round up.
向(距離)最近的一邊舍入,除非兩邊(的距離)是相等,如果是這樣,向上舍入, 1.55保留一位
小數結果為1.6
static int
ROUND_UNNECESSARY
Rounding mode to assert that the requested operation has an exact result, hence no
rounding is necessary.
計算結果是精確的,不需要舍入模式
static int
ROUND_UP
Rounding mode to round away from zero.
向遠離0的方向舍入
===========================================
如何應用Java的BigDecimal類
雙精度浮點型變量double可以處理16位有效數。在實際應用中,需要對更大或者更小的數進行運算和處理。
Java在java.math包中提供的API類BigDecimal,用來對超過16位有效位的數進行精確的運算。表5.7中列出了BigDecimal類的主要構造器和方法。
表5.7 BigDecimal類的主要構造器和方法
構造器  描 述
BigDecimal(int)創建一個具有參數所指定整數值的對象。
BigDecimal(double)創建一個具有參數所指定雙精度值的對象。
BigDecimal(long)創建一個具有參數所指定長整數值的對象。
BigDecimal(String)創建一個具有參數所指定以字符串表示的數值的對象。
續表
方 法描 述
add(BigDecimal)BigDecimal對象中的值相加,然后返回這個對象。
subtract(BigDecimal)BigDecimal對象中的值相減,然后返回這個對象。
multiply(BigDecimal)BigDecimal對象中的值相乘,然后返回這個對象。
divide(BigDecimal)BigDecimal對象中的值相除,然后返回這個對象。
toString()將BigDecimal對象的數值轉換成字符串。
doubleValue()將BigDecimal對象中的值以雙精度數返回。
floatValue()將BigDecimal對象中的值以單精度數返回。
longValue()將BigDecimal對象中的值以長整數返回。
intValue()將BigDecimal對象中的值以整數返回。
注意,由於一般數值類型,例如double,不能准確地代表16位有效數以上的數字,在使用BigDecimal時,應用BigDecimal(String)構造器創建對象才有意義。另外,BigDecimal所創建的是對象,我們不能使用傳統的+、-、*、/等算術運算符直接對其對象進行數學運算,而必須調用其相對應的方法。方法中的參數也必須是BigDecimal的對象。
構造器是類的特殊方法,專門用來創建對象,特別是帶有參數的對象。關於構造器概念和編寫技術,將在本書第6章詳細介紹。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM