Catching exception: divide by zero

后端 未结 8 1642
时光取名叫无心
时光取名叫无心 2020-11-22 16:23

The following code does not catch an exception, when I try to divide by 0. Do I need to throw an exception, or does the computer automatically throw one at runtime?

相关标签:
8条回答
  • 2020-11-22 17:22

    Updated with comments from ExcessPhase

    GCC (at least version 4.8) will let you emulate this behaviour:

    #include <signal.h>
    #include <memory>
    #include <iostream>
    
    int main() {
        std::shared_ptr<void(int)> handler(
            signal(SIGFPE, [](int signum) {throw std::logic_error("FPE"); }),
            [](__sighandler_t f) { signal(SIGFPE, f); });
    
        int i = 0;
    
        std::cin >> i;  // what if someone enters zero?
    
        try {
            i = 5/i;
        }
        catch (std::logic_error e) {
            std::cerr << e.what();
        }
    }
    

    This sets up a new signal handler which throws an exception, and a shared_ptr to the old signal handler, with a custom 'deletion' function that restores the old handler when it goes out of scope.

    You need to compile with at least these options:

    g++ -c Foo.cc -o Foo.o -fnon-call-exceptions -std=c++11
    

    Visual C++ will also let you do something similar:

    #include <eh.h>
    #include <memory>
    
    int main() {
        std::shared_ptr<void(unsigned, EXCEPTION_POINTERS*)> handler(
            _set_se_translator([](unsigned u, EXCEPTION_POINTERS* p) {
                switch(u) {
                    case FLT_DIVIDE_BY_ZERO:
                    case INT_DIVIDE_BY_ZERO:
                        throw std::logic_error("Divide by zero");
                        break;
                    ...
                    default:
                        throw std::logic_error("SEH exception");
                }
            }),
            [](_se_translator_function f) { _set_se_translator(f); });
    
        int i = 0;
    
        try {
            i = 5 / i;
        } catch(std::logic_error e) {
            std::cerr << e.what();
        }
    }
    

    And of course you can skip all the C++11-ishness of this and put them in a traditional RAII-managing struct.

    0 讨论(0)
  • 2020-11-22 17:23

    As far as I know C++ specifications does not mention anything about divide by zero exeption. I believe you need to do it yourself...

    Stroustrup says, in "The Design and Evolution of C++" (Addison Wesley, 1994), "low-level events, such as arithmetic overflows and divide by zero, are assumed to be handled by a dedicated lower-level mechanism rather than by exceptions. This enables C++ to match the behaviour of other languages when it comes to arithmetic. It also avoids the problems that occur on heavily pipelined architectures where events such as divide by zero are asynchronous."`

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