Арифметическое исключение в GDB, но я не делю на ноль?

Я получаю Floating point exception (core dumped) ошибка в моей C++ программе, и GDB показывает, что проблема в строке, которая выполняет деление по модулю:

Program received signal SIGFPE, Arithmetic exception.
[Switching to Thread 0x7ffff6804700 (LWP 13931)]
0x00000000004023e8 in CompExp::eval (this=0x7fffec000e40, currVal=0)
    at exp.cpp:55
55              return (r==0) ? 0 : l % r;

Линия защищает от деления на ноль, и моя обратная трассировка показывает следующее:

#0  0x00000000004023e8 in CompExp::eval (this=0x7fffec000e40, currVal=0)
    at exp.cpp:55
        l = -2147483648
        r = -1

Поскольку я знаю, что делю не на ноль, что еще может быть причиной исключения?

3 ответа

Итак, я выяснил, что стало причиной проблемы - арифметическое исключение может быть вызвано либо делением на ноль, либо переполнением целого числа со знаком, что и произошло здесь. Целые числа без знака требуются для переноса при переполнении; поведение для целых чисел со знаком не определено.

Измените код на следующий, чтобы не пытаться взять по модулю отрицательное число, которое не определено:

return (r<=0) ? 0 : l % r;

Чтобы вычислить такое выражение по модулю: -2147483648 % -1требуется разделение, которое в данном случае выглядит как 32-разрядное (думаю, l а также r определяются как int). Правильный результат такого разделения будет 2147483648, но это значение не может быть представлено в 32-битном формате, поэтому создается арифметическое исключение.

Другие вопросы по тегам