为什么数学函数Math.round(0.49999999999999994) 返回 1
本文于1944天之前发表,文中内容可能已经过时。
问题
通过下面的程序你可以看出来,对于任意一个比0.5略小的都是舍去小数向下取整,只有0.5是例外.
1 | for (int i = 10; i >= 0; i--) { |
输出为:
1 | 10.5 rounded is 11 |
_译者注:请看输出的最后两行,0.49999999999999994的输出为1,而0.49999999999999999的输出为0
我使用的版本是 Java 6 update 31
回答
总结
在 Java 6(或者之前的版本),round(x)是用floor(x+0.5)实现的.¹ 这是一个规范上的bug,恰恰是在这种病理条件下.²Java 7 不再使用这个有问题的实现了.
问题
0.5+0.49999999999999994 在double的精度下的结果是1
1 | static void print(double d) { |
这是因为0.49999999999999994的指数比0.5的指数小,所以当它们两个相加时,0.49999999999999994的原数就会发生移位,然后最小精度单位(unit of least precision)/最后置单位(unit of last place)相应的变大了.
解决方案
自从Java 7以来,OpenJDK(举个栗子)实现如下⁴:
1 | public static long round(double a) { |
- http://docs.oracle.com/javase/6/docs/api/java/lang/Math.html#round%28double%29
- http://bugs.java.com/bugdatabase/view_bug.do?bug_id=6430675 (credits to @SimonNickerson for finding this)
- http://docs.oracle.com/javase/7/docs/api/java/lang/Math.html#round%28double%29
- http://grepcode.com/file/repository.grepcode.com/java/root/jdk/openjdk/7u40-b43/java/lang/Math.java#Math.round%28double%29