java.math.RoundingMode是如何工作的?

时间:2011-08-19 16:14:01

标签: java rounding

我在舍入方面遇到了麻烦。具体来说,在阅读完所有的javadoc后,我期待以下代码:

int n = (integer between 0 and 9, included)
new BigDecimal(n + 0.555d).setScale(2, RoundingMode.HALF_UP).doubleValue()

返回n + 0.56。相反,这些是n04的返回值:

 new BigDecimal(0.555d).setScale(2, RoundingMode.HALF_UP).doubleValue()
 0.56
 new BigDecimal(1.555d).setScale(2, RoundingMode.HALF_UP).doubleValue()
 1.55
 new BigDecimal(2.555d).setScale(2, RoundingMode.HALF_UP).doubleValue()
 2.56
 new BigDecimal(3.555d).setScale(2, RoundingMode.HALF_UP).doubleValue()
 3.56
 new BigDecimal(4.555d).setScale(2, RoundingMode.HALF_UP).doubleValue()
 4.55

我也试图改变舍入模式:

int n = (integer between 0 and 9, included)
new BigDecimal(n + 0.555d).setScale(2, RoundingMode.HALF_DOWN).doubleValue()

期望n + 0.55成为每个n的结果。相反,返回值与前一个示例完全相同:

 new BigDecimal(0.555d).setScale(2, RoundingMode.HALF_DOWN).doubleValue()
 0.56
 new BigDecimal(1.555d).setScale(2, RoundingMode.HALF_DOWN).doubleValue()
 1.55
 new BigDecimal(2.555d).setScale(2, RoundingMode.HALF_DOWN).doubleValue()
 2.56
 new BigDecimal(3.555d).setScale(2, RoundingMode.HALF_DOWN).doubleValue()
 3.56
 new BigDecimal(4.555d).setScale(2, RoundingMode.HALF_DOWN).doubleValue()
 4.55

我错过了什么吗?

3 个答案:

答案 0 :(得分:13)

你遇到的问题是,double不是一个精确的表示,你可以根据这个不精确的数字进行舍入。

BigDecimal bd = new BigDecimal(1.555d);
System.out.println("bd=" + bd);
bd = bd.setScale(2, RoundingMode.HALF_UP);
System.out.println("after rounding bd=" + bd);
double d = bd.doubleValue();
System.out.println("after rounding d=" + d);

打印

bd=1.5549999999999999378275106209912337362766265869140625
after rounding bd=1.55
after rounding d=1.55

然而

BigDecimal bd = BigDecimal.valueOf(1.555d);
System.out.println("bd=" + bd);
bd = bd.setScale(2, RoundingMode.HALF_UP);
System.out.println("after rounding bd=" + bd);
double d = bd.doubleValue();
System.out.println("after rounding d=" + d);

打印

bd=1.555
after rounding bd=1.56
after rounding d=1.56

这是有效的,因为BigDecimal.valueOf会根据打印时出现的双倍进行一些额外的舍入。


但是,除非性能/简单性不是问题,否则我不会使用BigDecimal。

double d = 1.555d;
System.out.println("d=" + d);
d = roundToTwoPlaces(d);
System.out.println("after rounding d=" + d);

public static double roundToTwoPlaces(double d) {
    return ((long) (d < 0 ? d * 100 - 0.5 : d * 100 + 0.5)) / 100.0;
}

打印

d=1.555
after rounding d=1.56

有关详细信息,请Double your money again比较不同舍入方式的效果。

答案 1 :(得分:1)

0.555ddouble值。它可能略大于0.555或略小。

答案 2 :(得分:0)

舍入模式用于舍入十进制值。

您可以使用。

double DecimalValue = 3.1452;
BigDecimal decimal = new BigDecimal(DecimalValue).setScale(2, RoundingMode.DOWN);

=&GT;

 RoundingMode.CEILING

舍入模式向正无穷大舍入。对于  正值这个舍入模式表现为UP,for  负值为DOWN。 规则:x.round() >= x

RoundingMode.DOWN

舍入模式,其中值向四舍五入  零。 规则:x.round().abs() <= x.abs()

RoundingMode.DOWN

舍入模式向负无穷大舍入。对于  正值这个舍入模式表现为DOWN,for  负值为UP。 规则:x.round() <= x

RoundingMode.HALF_DOWN

舍入模式,其中值向着舍入  最近的邻居。通过四舍五入打破关系。

RoundingMode.HALF_EVEN

舍入模式,其中值向着舍入  最近的邻居。通过四舍五入来打破关系  邻居。

RoundingMode.HALF_UP

舍入模式,其中值向着舍入  最近的邻居。通过四舍五入打破关系。

 RoundingMode.UNNECESSARY

舍入操作抛出的舍入模式  对于四舍五入的情况,ArithmeticException  必要的,即对于价值不可能的情况  完全代表。

RoundingMode.UP

舍入模式,其中正值被舍入  朝向正无穷大和负值  负无穷大。 规则:x.round().abs() >= x.abs()