Double comparison in Java

Recently I was solving an interesting bug that came down to comparing two Double variables with equals method. It looks innocent, what can be wrong with something like firstDouble.equals(secondDouble)?
The problem here is with how doubles are stored. To fit them into 64bytes (usually) they are rounded.

See the example below:

Double firstDouble = 0d;
for (int i = 1; i <= 42; i++) {
 firstDouble += 0.1;
}

Double secondDouble = 0.1 * 42;

System.out.println(firstDouble); // 4.200000000000001
System.out.println(secondDouble); // 4.2
System.out.println(firstDouble.equals(secondDouble)); // false

Enter fullscreen mode Exit fullscreen mode

This inaccuracy is caused by rounding errors.
We need to use a different approach to compare those doubles.

Threshold method

If we do not have access to any libraries and want to solve this with Java only we can use something called the threshold method.
Simply, we will subtract those doubles, make absolute value, and compare if the result is smaller than some very small number.

double epsilon = 0.000001d;
boolean result = Math.abs(firstDouble - secondDouble) < epsilon;
System.out.println(result); // true

Enter fullscreen mode Exit fullscreen mode

That small number is called epsilon and the smaller it is the better the accuracy of the result. For most cases, 5 decimals should be enough.

Apache Commons Math

There is no utility method for this in JDK. Luckily for us, Apache Commons Math library has us covered. With it we can compare those doubles like so:

double epsilon = 0.000001d;
boolean result = Precision.equals(firstDouble, secondDouble, epsilon)
System.out.println(result);

Enter fullscreen mode Exit fullscreen mode

The epsilon has the same meaning as in the example above.

There are similar methods in Guava and other libraries.


You can follow me on Twitter to get more tips like this.

原文链接:Double comparison in Java

© 版权声明
THE END
喜欢就支持一下吧
点赞10 分享
评论 抢沙发

请登录后发表评论

    暂无评论内容