C++ warning: division of double by zero

12,117

Solution 1

Floating point division by zero is well defined by IEEE and gives infinity (either positive or negative according to the value of the numerator (or NaN for ±0) ).

For integers, there is no way to represent infinity and the language defines the operation to have undefined behaviour so the compiler helpfully tries to steer you clear from that path.

However in this case, since the numerator is a double, the divisor (0) should be promoted to a double too and there's no reason to give a warning here whilst not giving a warning for 0.0 so I think this is a compiler bug.

Solution 2

In Standard C++, both cases are undefined behaviour. Anything may happen, including formatting your hard drive. You should not expect or rely on "return inf. Ok" or any other behaviour.

The compiler apparently decides to give a warning in one case and not the other, but this doesn't mean that one code is OK and one isn't. It is just a quirk of the compiler's generation of warnings.

From the C++17 standard [expr.mul]/4:

The binary / operator yields the quotient, and the binary % operator yields the remainder from the division of the first expression by the second. If the second operand of / or % is zero the behavior is undefined.

Solution 3

My best guess to answer this particular question would be that compiler emits warning before performing conversion of int to double.

So, the steps would be like this:

  1. Parse expression
  2. Arithmetic operator /(T, T2), where T=double, T2=int.
  3. Check that std::is_integral<T2>::value is true and b == 0 - this triggers warning.
  4. Emit warning
  5. Perform implicit conversion of T2 to double
  6. Perform well defined division (since compiler decided to use IEEE 754).

This is of course speculation and is based on compiler-defined specifications. From standard point of view, we're dealing with possible Undefined Behaviours.


Please note that this is expected behaviour according to GCC documentation
(btw. it seems that this flag can't be used explicitly in GCC 8.1)

-Wdiv-by-zero
Warn about compile-time integer division by zero. This is default. To inhibit the warning messages, use -Wno-div-by-zero. Floating point division by zero is not warned about, as it can be a legitimate way of obtaining infinities and NaNs.

Solution 4

I will not go into the UB / not UB debacle in this answer.

I just want to point that 0 and 0.0 are different despite 0 == 0.0 evaluating to true. 0 is an int literal and 0.0 is a double literal.

However in this case the end result is the same: d/0 is floating point division because d is double and so 0 is implicitly converted to double.

Solution 5

I would argue that foo/0 and foo/0.0 are not the same. Namely, the resulting effect of the first (integer division or floating point division) is highly dependant on the type of foo, while the same is not true for the second (it will always be a floating point division).

Whether any of the two is UB is irrelevant. Quoting the standard:

Permissible undefined behavior ranges from ignoring the situation completely with unpredictable results, to behaving during translation or program execution in a documented manner characteristic of the environment (with or without the issuance of a diagnostic message), to terminating a translation or execution (with the issuance of a diagnostic message).

(Emphasis mine)

Consider the "suggest parentheses around assignment used as truth value" warning: The way to tell the compiler that you really want to use the result of an assignment is by being explicit, and adding parenthesis around the assignment. The resulting statement has the same effect, but it tells the compiler you know what you're doing. The same can be said about foo/0.0: Since you're explicitly telling the compiler "This is floating point division" by using 0.0 instead of 0, the compiler trusts you and will not issue a warning.

Share:
12,117
Jayesh
Author by

Jayesh

Updated on June 08, 2022

Comments

  • Jayesh
    Jayesh almost 2 years

    Case 1:

    #include <iostream>
    
    int main()
    {
        double d = 15.50;
        std::cout<<(d/0.0)<<std::endl;
    }
    

    It compiles without any warnings and prints inf. OK, C++ can handle division by zero, (see it live).

    But,

    Case 2:

    #include <iostream>
    
    int main()
    {
        double d = 15.50;
        std::cout<<(d/0)<<std::endl;
    }
    

    The compiler gives the following warning (see it live):

    warning: division by zero [-Wdiv-by-zero]
         std::cout<<(d/0)<<std::endl;
    

    Why does the compiler give a warning in the second case?

    Is 0 != 0.0?

    Edit:

    #include <iostream>
    
    int main()
    {
        if(0 == 0.0)
            std::cout<<"Same"<<std::endl;
        else
            std::cout<<"Not same"<<std::endl;
    }
    

    output:

    Same