Compare double to zero using epsilon

后端 未结 11 1294
醉梦人生
醉梦人生 2020-11-29 15:54

Today, I was looking through some C++ code (written by somebody else) and found this section:

double someValue = ...
if (someValue <  std::numeric_limits&         


        
相关标签:
11条回答
  • 2020-11-29 16:22

    An aproximation of epsilon (smallest possible difference) around a number (1.0, 0.0, ...) can be printed with the following program. It prints the following output:
    epsilon for 0.0 is 4.940656e-324
    epsilon for 1.0 is 2.220446e-16
    A little thinking makes it clear, that the epsilon gets smaller the more smaller the number is we use for looking at its epsilon-value, because the exponent can adjust to the size of that number.

    #include <stdio.h>
    #include <assert.h>
    double getEps (double m) {
      double approx=1.0;
      double lastApprox=0.0;
      while (m+approx!=m) {
        lastApprox=approx;
        approx/=2.0;
      }
      assert (lastApprox!=0);
      return lastApprox;
    }
    int main () {
      printf ("epsilon for 0.0 is %e\n", getEps (0.0));
      printf ("epsilon for 1.0 is %e\n", getEps (1.0));
      return 0;
    }
    
    0 讨论(0)
  • 2020-11-29 16:24

    The difference between X and the next value of X varies according to X.
    epsilon() is only the difference between 1 and the next value of 1.
    The difference between 0 and the next value of 0 is not epsilon().

    Instead you can use std::nextafter to compare a double value with 0 as the following:

    bool same(double a, double b)
    {
      return std::nextafter(a, std::numeric_limits<double>::lowest()) <= b
        && std::nextafter(a, std::numeric_limits<double>::max()) >= b;
    }
    
    double someValue = ...
    if (same (someValue, 0.0)) {
      someValue = 0.0;
    }
    
    0 讨论(0)
  • 2020-11-29 16:24

    So let's say system cannot distinguish 1.000000000000000000000 and 1.000000000000000000001. that is 1.0 and 1.0 + 1e-20. Do you think there still are some values that can be represented between -1e-20 and +1e-20?

    0 讨论(0)
  • 2020-11-29 16:34

    You can't apply this to 0, because of mantissa and exponent parts. Due to exponent you can store very little numbers, which are smaller than epsilon, but when you try to do something like (1.0 - "very small number") you'll get 1.0. Epsilon is an indicator not of value, but of value precision, which is in mantissa. It shows how many correct consequent decimal digits of number we can store.

    0 讨论(0)
  • 2020-11-29 16:35

    I think that depend on the precision of your computer. Take a look on this table: you can see that if your epsilon is represented by double, but your precision is higher, the comparison is not equivalent to

    someValue == 0.0
    

    Good question anyway!

    0 讨论(0)
提交回复
热议问题