考虑此代码:
import java.math.BigDecimal;
import java.math.RoundingMode;
public class RoundingTests {
public static void main(String[] args) {
BigDecimal bd1 = new BigDecimal(265.345d);
BigDecimal bd2 = new BigDecimal(265.335d);
System.out.println("Setting scale 265.345: " + bd1.setScale(2, RoundingMode.HALF_EVEN));
System.out.println("Setting scale 265.335: " + bd2.setScale(2, RoundingMode.HALF_EVEN));
}
}
输出为:
Setting scale 265.345: 265.35
Setting scale 265.335: 265.33
现在这与我所期望的完全相反。使用RoundingMode.HALF_EVEN
(也称为银行家舍入),我期望这两个值都将成为值265.34
请注意,我不是故意使用BigDecimal.round
方法,因为它也不是我所需要的。如果我添加此代码(并导入java.math.MathContext):
System.out.println("Rounding 265.345: " + bd1.round(new MathContext(2, RoundingMode.HALF_EVEN)));
System.out.println("Rounding 265.335: " + bd2.round(new MathContext(2, RoundingMode.HALF_EVEN)));
然后输出是:
Rounding 265.345: 2.7E+2
Rounding 265.335: 2.7E+2
这是预料之中的,并且在BigDecimal setScale and round中也进行了解释,但是对于我而言,这基本上是没有用的。
有人可以在这里解释setScale的问题吗?
更新:所以这只是另一个浮点问题,没有简单的方法来解决它,而不是从一开始就使用BigDecimals。
您可能希望将代码更改为使用BigDecimal.valueOf()
而不是new BigDecimal()
。
代码
Double dValue = 265.345d;
Double dValue2 = 265.335d;
BigDecimal value = BigDecimal.valueOf(dValue);
BigDecimal bd1 = new BigDecimal(265.345d);
BigDecimal bd2 = new BigDecimal(265.335d);
BigDecimal value2 = BigDecimal.valueOf(dValue2);
System.out.println("BigDecimal.valueOf(dValue);");
System.out.println(value.toPlainString());
System.out.println(String.valueOf(dValue));
value = value.setScale(2, BigDecimal.ROUND_HALF_EVEN);
System.out.println(value);
System.out.println("BigDecimal.valueOf(dValue2);");
System.out.println(value2.toPlainString());
System.out.println(String.valueOf(dValue2));
value2 = value2.setScale(2, BigDecimal.ROUND_HALF_EVEN);
System.out.println(value2);
System.out.println("BigDecimal bd1 = new BigDecimal(265.345d);");
System.out.println(bd1.setScale(2, BigDecimal.ROUND_HALF_EVEN));
System.out.println("BigDecimal bd2 = new BigDecimal(265.335d);");
System.out.println(bd2.setScale(2, BigDecimal.ROUND_HALF_EVEN));
输出:
BigDecimal.valueOf(dValue);
265.345
265.345
265.34
BigDecimal.valueOf(dValue2);
265.335
265.335
265.34
BigDecimal bd1 = new BigDecimal(265.345d);
265.35
BigDecimal value2 = BigDecimal.valueOf(dValue2);
265.33