通用程序设计——如果需要精确,避免使用float和double

float和double主要为了科学计算和工程计算而设计,执行二进制浮点运算,这是为了在广泛的数值范围上提供较为精确的快速近似计算而精心设计的。然而,它们没有提供完全精确的结果,所以不适合用于需要精确结果的场合,尤其是货币计算。

//假设有1.03元,花掉0.42元后
System.out.println(1.03 - 0.42);//0.6100000000000001

使用舍入可以解决上面问题,但并不是所有的问题都能用舍入解决,如有1元,有0.1,0.2,0.3,一直到1元的糖果,从0.1开始买,直到不能支付为止:

public static void main(String[] args) {
    double funds = 1.00;
    int itemsBought = 0;
    for(double price = .10; funds >= price; price += .10) {
        funds -= price;
        itemsBought++;
    }
    System.out.println(itemsBought + " items bought.");
    System.out.println("Money left over: $" + funds);
}

结果是:

3 items bought.
Money left over: $0.3999999999999999

使用BigDecimal代替double:

 public static void main(String[] args) {
        final BigDecimal TEN_CENTS = new BigDecimal("0.10");
        int itemsBought = 0;
        BigDecimal funds = new BigDecimal("1.00");
        for(BigDecimal price = TEN_CENTS; funds.compareTo(price) >= 0; price = price.add(TEN_CENTS)) {
            itemsBought++;
            funds = funds.subtract(price);
        }
        System.out.println(itemsBought + " items bought.");
        System.out.println("Money left over: $" + funds);
    }

 结果是:

4 items bought.
Money left over: $0.00

使用BigDecimal有2个缺点:1.与基本类型相比,不方便(需要创建BigDecimal对象);2.速度慢

除了使用BigDecimal,还可以使用int或者long,取决于涉及的数值大小,同时要自己处理十进制小数,以分为单位,而不是以元为单位计算,就可以使用int来处理:

public static void main(String[] args) {
    int itemsBought = 0;
    int funds = 100;
    for(int price = 10; funds >= price; price += 10) {
        itemsBought++;
        funds -= price;
    }
    System.out.println(itemsBought + " items bought.");
    System.out.println("Money left over: $" + funds);
}

 结果与用BigDecimal是一样的。

总结:

  1. 对于任何需要精确答案的计算任务,请不要使用float或者double。
  2. 如果你想让系统来记录十进制小数点,并且不介意因为不使用基本类型而带来的不便,就请使用BigDecimal。使用BigDecimal还有一些额外的好处,他允许你完全控制舍入,每当一个操作涉及舍入的时候,他允许你从8种舍入模式中选择其一。如果你正通过法定要求的舍入行为进行业务计算,使用BigDecimal是非常方便的。
  3. 如果性能非常关键,并且你不介意自己记录十进制小数点,而且所涉及的数值又不太大,就可以使用int或者long。如果数值范围内没有超过9位十进制数字,就可以使用int;如果不超过18位数字,就可以使用long。如果数值可能超过18位数字,就必须使用BigDecimal。

你可能感兴趣的:(java,effective,c++,开发语言)