JAVA 精确计算

示例代码

public class TestArithmetic {

    public static void main(String[] args) {
        System.out.println("0.09 + 0.01 = " + (0.09 + 0.01));
        System.out.println("9.99 - 0.12 = " + (9.99 - 0.12));
        System.out.println("17.173 * 100 = " + (17.173 * 100));
        System.out.println("16.3 / 1000 = " + (16.3 / 1000));
    }
}

运行结果

0.09 + 0.01 = 0.09999999999999999
9.99 - 0.12 = 9.870000000000001
17.173 * 100 = 1717.2999999999997
16.3 / 1000 = 0.016300000000000002

是不是怀疑以上结果写错了?但是结果确实是这样的,你可以亲自运行以上代码试试。

WHY?

为什么出现这种情况?因为 floatdouble 浮点类型在计算机中是无法准确表示的,例如 0.01 在计算机中只是表示成一个近似值,因此对于浮点数参与运算的结果会出现不准确的情况。

如何解决

使用 BigDecimal 进行精确计算

public class ArithmeticUtils {
    private static final int DEF_SCALE = 10;

    public static double add(double d1, double d2) {
        BigDecimal b1 = new BigDecimal(Double.toString(d1));
        BigDecimal b2 = new BigDecimal(Double.toString(d2));
        return b1.add(b2).doubleValue();
    }

    public static double subtract(double d1, double d2) {
        BigDecimal b1 = new BigDecimal(Double.toString(d1));
        BigDecimal b2 = new BigDecimal(Double.toString(d2));
        return b1.subtract(b2).doubleValue();
    }

    public static double multiply(double d1, double d2) {
        BigDecimal b1 = new BigDecimal(Double.toString(d1));
        BigDecimal b2 = new BigDecimal(Double.toString(d2));
        return b1.multiply(b2).doubleValue();
    }

    public static double divide(double d1, double d2) {
        return divide(d1, d2, DEF_SCALE);
    }

    public static double divide(double d1, double d2, int scale) {
        if (scale < 0) {
            throw new IllegalArgumentException("The scale must be a positive integer or zero");
        }
        BigDecimal b1 = new BigDecimal(Double.toString(d1));
        BigDecimal b2 = new BigDecimal(Double.toString(d2));
        return b1.divide(b2, scale, BigDecimal.ROUND_HALF_UP).doubleValue();
    }
}

再次运行以下代码

public class TestArithmetic {

    public static void main(String[] args) {
        System.out.println("0.09 + 0.01 = " + ArithmeticUtils.add(0.09, 0.01));
        System.out.println("9.99 - 0.12 = " + ArithmeticUtils.subtract(9.99, 0.12));
        System.out.println("17.173 * 100 = " + ArithmeticUtils.multiply(17.173, 100));
        System.out.println("16.3 / 1000 = " + ArithmeticUtils.divide(16.3, 1000));
    }
}

正确结果输出

0.09 + 0.01 = 0.1
9.99 - 0.12 = 9.87
17.173 * 100 = 1717.3
16.3 / 1000 = 0.0163

转载于:https://my.oschina.net/XngPro/blog/841163

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值