一尘不染

Java-四舍五入到小数点后两位

java

如果值为200.3456,则应将其格式化为200.34。如果是200,那么应该是200.00


阅读 385

收藏
2020-02-25

共1个答案

一尘不染

这是一个实用程序,可以将双精度舍入(而不是截断)双精度至小数位数。

例如:

round(200.3456, 2); // returns 200.35

原始版本;提防这个

public static double round(double value, int places) {
    if (places < 0) throw new IllegalArgumentException();

    long factor = (long) Math.pow(10, places);
    value = value * factor;
    long tmp = Math.round(value);
    return (double) tmp / factor;
}

这在小数位数很高(例如)或整数部分很大(例如)的极端情况下会严重崩溃。感谢Sloin指出这一点。round(1000.0d, 17)round(90080070060.1d, 9)

多年来,我一直在使用上面的方法将“不太大”的双精度取整到2或3个小数位(例如,以秒为单位的清理时间用于记录目的:27.987654321987-> 27.99)。但是我想最好避免使用它,因为更可靠的方法已经可以使用,并且代码也更简洁。

因此,改用它
(改编自Louis Wasserman的回答和Sean Owen的回答。)

public static double round(double value, int places) {
    if (places < 0) throw new IllegalArgumentException();

    BigDecimal bd = BigDecimal.valueOf(value);
    bd = bd.setScale(places, RoundingMode.HALF_UP);
    return bd.doubleValue();
}

注意,这HALF_UP是“学校常用”的舍入模式。细读的RoundingMode文件,如果你怀疑你需要别的东西,如银行家舍入。

当然,如果你愿意,可以将以上内容内联为单行:

new BigDecimal(value).setScale(places, RoundingMode.HALF_UP).doubleValue()

而且在每种情况下

永远记住,浮点表示使用float和double是不精确的。例如,考虑以下表达式:

999199.1231231235 == 999199.1231231236 // true
1.03 - 0.41 // 0.6200000000000001

为了精确起见,你想使用BigDecimal。同时,使用带String的构造函数,而不要带double的构造函数。例如,尝试执行以下命令:

System.out.println(new BigDecimal(1.03).subtract(new BigDecimal(0.41)));
System.out.println(new BigDecimal("1.03").subtract(new BigDecimal("0.41")));

关于该主题的一些出色的进一步阅读:

  • 项目48:Joshua Bloch撰写的Effective Java(第二版)中的“避免float并且double如果需要确切的答案”
  • 每个程序员应了解的浮点运算法则
    如果你希望使用字符串格式而不是(或除了)严格舍入数字,请参阅其他答案。

具体来说,请注意round(200, 0)return 200.0。如果要输出“ 200.00 ”,则应首先取整,然后格式化输出的结果(这在Jesper的答案中有很好的解释)。

2020-02-25