`

Java中BigDecimal精度转换问题

    博客分类:
  • Java
 
阅读更多

[1] 精确的浮点运算: 
在Java里面,有时候为了保证数值的准确性需要精确的数据,先提供一个例子就可以发现问题了: 

Java代码 
  1. public class FloatNumberTester {  
  2.     public static void main(String args[]){  
  3.         System.out.println(0.05+0.01);  
  4.         System.out.println(1.0 - 0.42);  
  5.         System.out.println(4.015 * 100);  
  6.         System.out.println(123.3 / 100);  
  7.     }  
  8. }  


按照我们的期待,上边应该是什么结果呢,但是看输出我们就会发现问题了: 

Java代码 
  1. 0.060000000000000005  
  2. 0.5800000000000001  
  3. 401.49999999999994  
  4. 1.2329999999999999  


这样的话这个问题就相对严重了,如果我们使用123.3元交易,计算机却因为1.2329999999999999而拒绝了交易,岂不是和实际情况大相径庭。 

[2] 四舍五入: 
另外的一个计算问题,就是四舍五入。但是Java的计算本身是不能够支持四舍五入的,比如: 

Java代码 
  1. public class GetThrowTester {  
  2.     public static void main(String args[]){  
  3.         System.out.println(4.015 * 100.0);  
  4.     }  
  5. }  

  
这个输出为: 
401.49999999999994 
所以就会发现这种情况并不能保证四舍五入,如果要四舍五入,只有一种方法 
java.text.DecimalFormat: 

Java代码 
  1. import java.text.DecimalFormat;  
  2. public class NumberFormatMain {  
  3.     public static void main(String args[]){  
  4.         System.out.println(new DecimalFormat("0.00").format(4.025));  
  5.         System.out.println(new DecimalFormat("0.00").format(4.024));  
  6.     }  
  7. }  


上边代码输出为: 

Java代码
  1. 4.02   
  2. 4.02   


发现问题了么?因为DecimalFormat使用的舍入模式, 舍入模式 详情参见本文最后部分。 
[3] 浮点输出: 
  Java浮点类型数值在大于9999999.0就自动转化成为科学计数法,看看下边的例子: 

Java代码
  1.    
  2. public class FloatCounter {  
  3.     public static void main(String args[]){  
  4.         System.out.println(9969999999.04);  
  5.         System.out.println(199999999.04);  
  6.         System.out.println(1000000011.01);  
  7.         System.out.println(9999999.04);  
  8.     }  
  9. }  


输出结果为: 

Java代码 
  1.    
  2. 9.96999999904E9   
  3. 1.9999999904E8   
  4. 1.00000001101E9   
  5. 9999999.04  
  6.    


但是有时候我们不需要科学计数法,而是转换成为字符串,所以这样可能会有点麻烦。 
总结: 
所以在项目当中,对于浮点类型以及大整数的运算 还是尽量不要用double,long等基本数据类型以及其包装类,还是用Java中提供的BigDecimal,BigInteger等大数值类型来代替吧。 
但这里特别说明一下BigDecimal类的两个构造函数的区别,他们分别是: 
new BigDecimal(String  val ) 和 new BigDecimal(double  val ) 
先看例子: 

Java代码
  1. public class BigDecimalMain {  
  2.     public static void main(String args[]){  
  3.         System.out.println(new BigDecimal(123456789.01).toString());  
  4.         System.out.println(new BigDecimal("123456789.01").toString());  
  5.     }  
  6. }  


输出结果有一次令人意外了,同时两者之间的区别也一目了然了: 

Java代码 
  1. 123456789.01000000536441802978515625   
  2. 123456789.01   


所以在 就是想利用double原始类型进行了相关计算之后再转成BigDecimal类型 的场合下,为了防止精度出现偏离,建议使用参数为String类型的该构造方法。即new BigDecimal(String  val )。 


BigDecimal舍入模式介绍: 
  舍入模式在java.math.RoundingMode 里面: 
RoundingMode.CEILING :向正无限大方向舍入的舍入模式。如果结果为正,则舍入行为类似于 RoundingMode.UP;如果结果为负,则舍入行为类似于 RoundingMode.DOWN。注意,此舍入模式始终不会减少计算值 
 

输入数字 使用CEILING舍入模式将数字舍入为一位数 
5.5
2.5
1.1
1.0
-1.0 -1 
-1.1 -1 
-1.6 -1 
-2.5 -2
-5.5 -5


RoundingMode.DOWN :向零方向舍入的舍入模式。从不对舍弃部分前面的数字加 1(即截尾)。注意,此舍入模式始终不会增加计算值的绝对值 

输入数字 使用DOWN舍入模式将数字舍入为一位数 
5.5
2.5
1.1
-1.0 -1 
-1.6 -1 
-2.5 -2 
-5.5 -5 


RoundingMode.FLOOR :向负无限大方向舍入的舍入模式。如果结果为正,则舍入行为类似于 RoundingMode.DOWN;如果结果为负,则舍入行为类似于 RoundingMode.UP。注意,此舍入模式始终不会增加计算值  

输入数字 使用FLOOR舍入模式将输入数字舍入为一位 
5.5
2.3
1.6
1.0
-1.1 -2 
-2.5 -3 
-5.5 -6



RoundingMode.HALF_DOWN :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向下舍入。如果被舍弃部分 > 0.5,则舍入行为同 RoundingMode.UP;否则舍入行为同 RoundingMode.DOWN   

输入数字 使用HALF_DOWN输入模式舍入为一位 
5.5
2.5
1.6
1.0
-1.1 -1 
-1.6 -2 
-2.5 -2 
-5.5 -5 


RoundingMode.HALF_EVEN :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向相邻的偶数舍入。如果舍弃部分左边的数字为奇数,则舍入行为同 RoundingMode.HALF_UP;如果为偶数,则舍入行为同 RoundingMode.HALF_DOWN。注意,在重复进行一系列计算时,此舍入模式可以在统计上将累加错误减到最小。此舍入模式也称为“银行家舍入法”,主要在美国使用。此舍入模式类似于 Java 中对 float 和 double 算法使用的舍入策略 

输入数字 使用HALF_EVEN舍入模式将输入舍为一位 
5.5
2.5
1.6
1.1
-1.0 -1 
-1.6 -2 
-2.5 -2 
-5.5 -6 


RoundingMode.HALF_UP :向最接近数字方向舍入的舍入模式,如果与两个相邻数字的距离相等,则向上舍入。如果被舍弃部分 >= 0.5,则舍入行为同 RoundingMode.UP;否则舍入行为同 RoundingMode.DOWN。注意,此舍入模式就是通常学校里讲的四舍五入  

输入数字 使用HALF_UP舍入模式舍入为一位数 
5.5
2.5
1.6
1.0
-1.1 -1 
-1.6 -2 
-2.5 -3 
-5.5 -6 



RoundingMode.UNNECESSARY :用于断言请求的操作具有精确结果的舍入模式,因此不需要舍入。如果对生成精确结果的操作指定此舍入模式,则抛出 ArithmeticException 

输入数字 使用UNNECESSARY模式 
5.5 抛出 ArithmeticException 
2.5 抛出 ArithmeticException 
1.6 抛出 ArithmeticException 
1.0
-1.0 -1.0 
-1.1 抛出 ArithmeticException 
-1.6 抛出 ArithmeticException 
-2.5 抛出 ArithmeticException 
-5.5 抛出 ArithmeticException



RoundingMode.UP :远离零方向舍入的舍入模式。始终对非零舍弃部分前面的数字加 1。注意,此舍入模式始终不会减少计算值的绝对值 

输入数字 使用UP舍入模式将输入数字舍入为一位数 
5.5
1.6
1.1
1.0
-1.1 -2 
-1.6 -2 
-2.5 -3 
-5.4 -6




Java代码 
  1. import  java.math.BigDecimal;   
  2. import  java.text.DecimalFormat;   
  3. /**  
  4.  *使用舍入模式的格式化操作  
  5.  **/   
  6. public class   DoubleFormat {   
  7.     public static void  main(String  args[]){   
  8.         DoubleFormat format =  new  DoubleFormat();   
  9.         System.out .println(format.doubleOutPut(12.3452));   
  10.         System.out .println(format.roundNumber(12.3352));   
  11.     }   
  12.     public   String  doubleOutPut(double  v,Integer num){   
  13.         if ( v == Double.valueOf(v).intValue()){   
  14.             return  Double.valueOf(v).intValue() +  "" ;   
  15.         }else {   
  16.             BigDecimal b =  new  BigDecimal(Double.toString(v));   
  17.             return  b.setScale(num,BigDecimal.ROUND_HALF_UP ).toString();   
  18.         }   
  19.     }   
  20.     public   String  roundNumber(double  v,int  num){   
  21.         String  fmtString =  "0000000000000000" ;  //16bit   
  22.         fmtString = num>0 ?  "0."   + fmtString.substring(0,num):"0" ;   
  23.         DecimalFormat dFormat =  new  DecimalFormat(fmtString);   
  24.         return  dFormat.format(v);   
  25.     }   
  26. }   

  
 这段代码的输出为: 

Java代码 
  1. 12.35   
  2. 12.34  
  3.    



本文转自:http://tidercreverse.group.iteye.com/group/topic/25085

分享到:
评论

相关推荐

    java中BigDecimal的操作方法

    在Java编程语言中,BigDecimal是用于处理高精度和可配置精度的十进制数的类。在进行商业计算时,由于浮点数(double和float)存在精度问题,不能保证准确的结果,因此通常推荐使用BigDecimal来确保计算的精确性。本文...

    BigDecimal向Double转换

    在Java编程语言中,BigDecimal是一种用于处理高精度数值的数据类型,尤其适用于金融计算等领域,因为它可以提供不受限的小数位数精度以及精确的数学运算能力。然而,在某些情况下,我们可能需要将BigDecimal类型的值...

    java BigDecimal操作

    总的来说,理解和熟练运用BigDecimal是每个Java开发者必备的技能之一,特别是在处理货币计算、财务计算等场景时,能够避免因为浮点数精度问题导致的错误。在使用BigDecimal时,一定要注意其性能开销,因为它比基本的...

    将 BigDecimal 类型转换为 Integer 类型.pdf

    在 Java 编程环境中,数据类型的转换是常见的操作,特别是在处理不同精度和范围的数值时。BigDecimal 类型是 Java 提供的一种用于精确算术运算的类,它支持任意大小的十进制数,并且能够避免浮点数计算时可能出现的...

    Java中BigDecimal精度和相等比较的坑

    在Java编程中,`BigDecimal` 类是用来处理高精度、任意精度的十进制数的,尤其适用于财务和货币计算,因为它能确保计算结果的精确性。`float` 和 `double` 类型虽然在科学计算和工程计算中广泛使用,但它们无法提供...

    Java中BigDecimal的加减乘除、比较大小与使用注意事项

    7. **避免自动装箱拆箱**:尽量避免`BigDecimal`与原始类型的直接转换,这可能导致精度损失。 8. **运算效率**:连续多次使用`add()`、`subtract()`等方法可能会导致大量中间对象的生成,可以考虑使用`BigDecimal`...

    Java中BigDecimal类的使用详解

    由于浮点数的精度问题,Java中浮点数的计算会失去一定的精确度。因此,使用BigDecimal类可以避免浮点数的精度问题,进行精确的数学运算。 一、BigDecimal转换取Double数据 在使用BigDecimal类时,我们需要将Double...

    Java中BigDecimal的基本运算(详解)

    Java中的BigDecimal是一种高精度的数据类型,它可以用来表示非常大的整数和小数,提供了丰富的数学运算功能。下面我们将对Java中BigDecimal的基本运算进行详细的介绍。 构造方法 BigDecimal有四种构造方法,但是...

    Java Double 精度问题总结

    本文将详细介绍Java中的 `double` 类型精度问题,并提供几种解决方法。 #### 一、Java Double 精度丢失的原因 1. **二进制表示限制**:`double` 类型基于IEEE 754标准的双精度格式,使用64位来表示一个浮点数。...

    Java编程BigDecimal用法实例分享

    Java中的BigDecimal类是Java标准库中的一部分,用于处理高精度的数字计算,特别是在商业计算中对数字精度要求较高的场景中。BigDecimal类提供了对大数字的操作,可以精确地计算货币值等。 BigDecimal类的实现是基于...

    bigdecimal转integer.docx

    `BigDecimal` 用于处理精确的浮点数运算,适合财务或金融计算,因为它可以避免浮点数计算中的精度问题。而 `Integer` 是 Java 中的整数类型,它只能存储整数值。在某些情况下,我们可能需要将 `BigDecimal` 对象转换...

    Java对BigDecimal常用方法的归类(加减乘除).doc

    Java 中 BigDecimal 的常用方法归类(加减乘除) Java 中的 BigDecimal 类提供了对浮点数的精确运算,包括加减乘除和四舍五入等操作。在 Java 中,简单类型不能够精确地对浮点数进行运算,因此需要使用 BigDecimal ...

    浅谈java中BigDecimal类的简单用法

    在Java编程语言中,`BigDecimal` 类位于 `java.math`...总的来说,`BigDecimal` 类是Java中处理高精度计算的重要工具,理解其构造函数和方法的用法,以及如何避免精度问题,对于编写准确的财务或科学计算程序至关重要。

    Java小数精度、时间格式和类型转换.doc

    Java 小数精度、时间格式和类型转换 Java 中的类型转换是非常重要的,特别是在处理小数和日期时。在 Java 中,存在多种类型的转换,包括基本类型的转换、包装...小数精度问题和日期格式化是 Java 中非常重要的概念。

    java中BigDecimal进行加减乘除的基本用法

    在Java编程语言中,`BigDecimal` 类位于 `java.math` 包下,它提供了一种进行高精度和任意精度的十进制算术运算的方法。`BigDecimal` 是为那些需要精确数值计算的场景设计的,比如金融计算或会计应用,因为普通的 `...

    java.math.BigDecimal 操作类

    java.math.BigDecimal 操作类,包含加减乘除、String型加减乘除精度格式化转换计算等

    Java BigDecimal和double示例及相关问题解析

    在 Java 中,浮点数的表示形式有多种,包括 float 和 double 等,但是这些类型在进行计算时会出现精度问题,而 BigDecimal 则可以解决这种问题。本文将主要介绍 Java 中的 BigDecimal 和 double 的示例及相关问题...

    Java BigDecimal类用法详解

    - `BigDecimal(double val)`:使用double值构造BigDecimal时,由于double的精度问题,可能会导致非预期的结果,如示例中的1.22被转换成了1.219999999999999973。 - `BigDecimal(String val)`:推荐使用此构造方法...

    BigDecimal类

    BigDecimal 类是 Java 中的一种数值类型,主要用于处理超过 16 位有效数字的数值运算。该类提供了多种构造器和方法,用于创建和操作 BigDecimal 对象。 构造器 BigDecimal 类提供了四种构造器,用于创建不同的 ...

    javascript版BigDecimal类库

    为了解决这个问题,开发者们引入了`BigDecimal`类库的概念,它在Java中被广泛使用,用于进行高精度的算术运算。本文将详细介绍JavaScript版的`BigDecimal`类库,以及如何在JavaScript环境中实现精确计算。 ...

Global site tag (gtag.js) - Google Analytics