Java 中浮点数---------BigDecimal和double(初探)
为什么要使用 bigdecimal?
借用《Effactive Java》这本书中的话,float和double类型的主要设计目标是为了科学计算和工程计算。他们执行二进制浮点运算,这是为了在广域数值范围上提供较为精确的快速近似计算而精心设计的。然而,它们没有提供完全精确的结果,所以不应该被用于要求精确结果的场合。但是,商业计算往往要求结果精确,这时候BigDecimal就派上大用场啦。
BigDecimal简介
BigDecimal 由任意精度的整数非标度值 和32 位的整数标度 (scale) 组成。如果为零或正数,则标度是小数点后的位数。如果为负数,则将该数的非标度值乘以 10 的负scale 次幂。因此,BigDecimal表示的数值是(unscaledValue × 10-scale)。
构造方法
BigDecimal aString = new BigDecimal("1.22");
BigDecimal aDouble = new BigDecimal(1.22);
BigDecimal bDouble = BigDecimal.valueOf(1.22);
System.out.println(aString);
System.out.println(aDouble);
System.out.println(bDouble);
Result:
1.22
1.2199999999999999733546474089962430298328399658203125
1.22
源码分析:
BigDecimal.valueOf(1.22) 构造方法,其实是调用 构造方法BigDecimal(String val);
Jdk中有说明:
1、参数类型为double的构造方法的结果有一定的不可预知性。有人可能认为在Java中写入newBigDecimal(0.1)所创建的BigDecimal正好等于 0.1(非标度值 1,其标度为 1),但是它实际上等于0.1000000000000000055511151231257827021181583404541015625。这是因为0.1无法准确地表示为 double(或者说对于该情况,不能表示为任何有限长度的二进制小数)。这样,传入到构造方法的值不会正好等于 0.1(虽然表面上等于该值)。
2、另一方面,String 构造方法是完全可预知的:写入 newBigDecimal("0.1") 将创建一个 BigDecimal,它正好等于预期的 0.1。因此,比较而言,通常建议优先使用String构造方法。
3、当double必须用作BigDecimal的源时,请注意,此构造方法提供了一个准确转换;它不提供与以下操作相同的结果:先使用Double.toString(double)方法,然后使用BigDecimal(String)构造方法,将double转换为String。要获取该结果,请使用static valueOf(double)方法。
在数据量精度不是很大的情况下,通过decimal输出正确的结果。(保留三位小数)
BigDecimal aString = new BigDecimal("1.22");
BigDecimal aDouble = new BigDecimal(1.22);
BigDecimal aValue = BigDecimal.valueOf(1.22);
System.out.println(new DecimalFormat("0.000").format(aString));
System.out.println(new DecimalFormat("0.000").format(aDouble));
System.out.println(new DecimalFormat("0.000").format(aValue));
Result:
1.220
1.220
1.220
原因分析:DecimalFormat做用了自己的舍余规则
总结
(1)商业计算使用BigDecimal。
(2)尽量使用参数类型为String的构造函数。
(3) BigDecimal都是不可变的(immutable)的,在进行每一步运算时,都会产生一个新的对象,所以在做加减乘除运算时千万要保存操作后的值。
(4)我们往往容易忽略JDK底层的一些实现细节,导致出现错误,需要多加注意。
问题一:BigDecimal的精度问题(StackOverflow上有个家伙问了相关的问题)
System.out.println(new BigDecimal(0.1).toString()); // 0.1000000000000000055511151231257827021181583404541015625 System.out.println(new BigDecimal("0.1").toString()); // 0.1 System.out.println(new BigDecimal( Double.toString(0.1000000000000000055511151231257827021181583404541015625)).toString());// 0.1 System.out.println(new BigDecimal(Double.toString(0.1)).toString()); // 0.1
分析一下上面代码的问题(注释的内容表示此语句的输出)
int x=(int)1023.99999999999999; // x=1024为什么?
原因还是在于二进制无法精确地表示某些十进制小数,因此1023.99999999999999在编译之后的double值变成了1024。
double d = 1023.99999999999999; int x = (int) d; System.out.println(new BigDecimal(d).toString()); // 1024 System.out.println(Long.toHexString(Double.doubleToRawLongBits(d))); // 4090000000000000 System.out.println(x); // 1024
前面提过BigDecimal可以精确地把double表示出来还记得吧。
public static void main(String[] args) { BigDecimal a = BigDecimal.valueOf(1.01); BigDecimal b = BigDecimal.valueOf(1.00000001); if (a.compareTo(b) == 0) { System.out.println("a==b"); } else if (a.compareTo(b) == 1) { System.out.println("a>b"); } else if (a.compareTo(b) == -1) { System.out.println("a<b"); } }
参考:
http://www.cnblogs.com/mingforyou/p/3344489.html
http://stackoverflow.com/questions/8073912/why-do-we-need-to-convert-the-double-into-a-string-before-we-can-convert-it-int
http://www.cnblogs.com/linjiqin/p/3413894.html