My Profile Photo

Kris Lee


专注移动互联,提供移动开发、培训与解决方案


关于简单类型不能够精确的对浮点数进行运算的问题

    今天早上看到一帖子:     <永远不要相信计算机></a>

    里面说几个很明显的简单的浮点的运算,计算机都会算错。     我引过来给大家看看:     运行代码:

	System.out.println(0.05 + 0.01);
        System.out.println(1.0 - 0.42);
        System.out.println(4.015 * 100);
        System.out.println(123.3 / 100);
		

    大家可以自己新建一个工程来试试,结果如下:

0.060000000000000005
0.5800000000000001
401.49999999999994
1.2329999999999999

    这就很神奇了,0.05+0.01很明显是0.06嘛,但是为什么会变成0.060000000000000005?     是不是计算器太弱智?真是计算机就不靠谱了?

    当然不是的。主要是因为java中的简单类型并不适用于对浮点的精确计算。不光是JAVA ,其它语言也存在同样的问题。 虽然现在CPU都支持浮点的运算了,但是CPU在处理的时候,也是先把浮点数(float , double)转成整数再转成二进制,然后进行操作,如果有取余,会有不同的取余方式。 再加上运算完成后,再多二进制转成上层的浮点,又会有一些取舍。就造成了呈现出来时的简单明显的错误 。

    所以说,一般float和double用来做科学计算或者是工程计算,在一般对精度要求较高的地方(如商业),我们会用到BCD码或者是java.math.BigDecimal。

    BSD码(Binary-Coded Decimal),称BCD码或二-十进制代码,亦称二进码十进数。是一种二进制的数字编码形式,用二进制编码的十进制代码。 大家对这个有兴趣的可以深入研究一下,今天我们主要讲的是BigDecimal类。因为我们在做项目的时候,尤其是对商业项目时,我们不可能还去搞个什么BSD码,可以直接利用BigDecimal类来完成我们的需求。

    BigDecimal 是Java提供的不可变的、任意精度的有符号十进制数。如果想看更多关于BigDecimal的介绍,大家可以自行去查看JDK的文档。

    BigDecimal提供了一系列的构造函数,主用于将double , string等转化成BigDecimal对象。

BigDecimal(double val) 
	将 double 转换为 BigDecimal,后者是 double 的二进制浮点值准确的十进制表示形式。
BigDecimal(String val) 
    将 BigDecimal 的字符串表示形式转换为 BigDecimal。
	

    习惯上我们在使用浮点数的时候都是直接定义的double , float数据类型,在定义上是没有问题,但是如果我们直接调用BigDecimal(double val) 方法来转化,那我们可得先注意一下JDK文档中关于这个构造的详细说明了:

    直接上中文了,英文好的,可以自己去看原版:

注:

    此构造方法的结果有一定的不可预知性。有人可能认为在 Java 中写入 new BigDecimal(0.1) 所创建的 BigDecimal 正好等于 0.1(非标度值 1,其标度为 1),但是它实际上等于 0.1000000000000000055511151231257827021181583404541015625。这是因为 0.1 无法准确地表示为 double(或者说对于该情况,不能表示为任何有限长度的二进制小数)。这样,传入 到构造方法的值不会正好等于 0.1(虽然表面上等于该值)。

    BigDecimal(double val) 这个方法是不可预知的,所以我们推荐使用BigDecimal(String val) 。     String的构造函数就是可预知的,new BigDecimal(“.1”)如同期望的那样精确的等于.1。

    接下来我们对 0.05+0.01重新修改一下:

BigDecimal bd1 = new BigDecimal(Double.toString(0.05));
BigDecimal bd2 = new BigDecimal(Double.toString(0.01));
System.out.println(bd1.add( bd2));
		

    我们再来看看运行的结果:

0.06

    这下就是我们想要的结果了,完成了高精度的一个运算了。

注意:     现在我们已经知道怎么样来解决这个问题了,原则上是推荐使用BigDecimal(String val) 构造方法。     我建议,在商业的应用中,涉及到money的浮点运算全都定义成String ,在数据库中保存也是String ,在需要使用到这个money来作运算的时候,我们再把String转化成BigDecimal来完成高精度的运算。

    试想一下,如果我们要做一个加法运算,需要先将两个浮点数转为String,然后够造成BigDecimal,在其中一个上调用add方法,传入另一个作为参数,然后把运算的结果(BigDecimal)再转换为浮点数。你能够忍受这么烦琐的过程吗?

    SO, 网上找了一个比较好的一个工具类,封闭了简单的操作。可以参考一下:

/**
 * 
 * 由于Java的简单类型不能够精确的对浮点数进行运算,这个工具类提供精
 * 确的浮点数运算,包括加减乘除和四舍五入。
 * 
 *
 */
public class Arith{
    //默认除法运算精度
    private static final int DEF_SCALE = 10;
    //这个类不能实例化
    private Arith(){
    }
 
    /**
     * 提供精确的加法运算。
     * @param v1 被加数
     * @param v2 加数
     * @return 两个参数的和
     */
    public static double add(double v1,double v2){
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.add(b2).doubleValue();
    }
    /**
     * 提供精确的减法运算。
     * @param v1 被减数
     * @param v2 减数
     * @return 两个参数的差
     */
    public static double sub(double v1,double v2){
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.subtract(b2).doubleValue();
    } 
    /**
     * 提供精确的乘法运算。
     * @param v1 被乘数
     * @param v2 乘数
     * @return 两个参数的积
     */
    public static double mul(double v1,double v2){
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.multiply(b2).doubleValue();
    }
 
    /**
     * 提供(相对)精确的除法运算,当发生除不尽的情况时,精确到
     * 小数点以后10位,以后的数字四舍五入。
     * @param v1 被除数
     * @param v2 除数
     * @return 两个参数的商
     */
    public static double div(double v1,double v2){
        return div(v1,v2,DEF_SCALE);
    }
 
    /**
     * 提供(相对)精确的除法运算。当发生除不尽的情况时,由scale参数指
     * 定精度,以后的数字四舍五入。
     * @param v1 被除数
     * @param v2 除数
     * @param scale 表示表示需要精确到小数点以后几位。
     * @return 两个参数的商
     */
    public static double div(double v1,double v2,int scale){
        if(scale<0){
            throw new IllegalArgumentException(
                "The scale must be a positive integer or zero");
        }
        BigDecimal b1 = new BigDecimal(Double.toString(v1));
        BigDecimal b2 = new BigDecimal(Double.toString(v2));
        return b1.divide(b2,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
    }
 
    /**
     * 提供精确的小数位四舍五入处理。
     * @param v 需要四舍五入的数字
     * @param scale 小数点后保留几位
     * @return 四舍五入后的结果
     */
    public static double round(double v,int scale){
        if(scale<0){
            throw new IllegalArgumentException(
                "The scale must be a positive integer or zero");
        }
        BigDecimal b = new BigDecimal(Double.toString(v));
        BigDecimal one = new BigDecimal("1");
        return b.divide(one,scale,BigDecimal.ROUND_HALF_UP).doubleValue();
    }
}