Работа с исключениями с плавающей точкой
Я не уверен, что делать с исключениями с плавающей запятой в C или C++. Из вики есть следующие типы исключений с плавающей точкой:
IEEE 754 specifies five arithmetic errors that are to be recorded in "sticky bits" (by default; note that trapping and other alternatives are optional and, if provided, non-default). * inexact, set if the rounded (and returned) value is different from the mathematically exact result of the operation. * underflow, set if the rounded value is tiny (as specified in IEEE 754) and inexact (or maybe limited to if it has denormalisation loss, as per the 1984 version of IEEE 754), returning a subnormal value (including the zeroes). * overflow, set if the absolute value of the rounded value is too large to be represented (an infinity or maximal finite value is returned, depending on which rounding is used). * divide-by-zero, set if the result is infinite given finite operands (returning an infinity, either +∞ or −∞). * invalid, set if a real-valued result cannot be returned (like for sqrt(−1), or 0/0), returning a quiet NaN.
Возможно ли, что когда произойдет какой-либо из приведенных выше исключений, программа выйдет ненормально? Или программа выдаст эту ошибку, не упоминая ничего, и, следовательно, затруднит ее отладку?
Может ли такой компилятор, как gcc, выдавать предупреждение для какого-то очевидного случая?
Что я могу сделать во время кодирования моей программы, чтобы уведомить, где происходит ошибка, и какие это типы, когда она возникает, чтобы я мог легко найти ошибку в своем коде? Пожалуйста, дайте решения как в случае C, так и C++.
Спасибо и всего наилучшего!
6 ответов
В Linux вы можете использовать расширение GNU feenableexcept (скрытое справа внизу этой страницы) для включения перехвата исключений с плавающей запятой - если вы сделаете это, вы получите сигнал SIGFPE, когда возникнет исключение, которое вы можете затем перехватить в твой отладчик. Однако будьте осторожны, так как иногда сигнал передается по команде с плавающей запятой после той, которая на самом деле вызывает проблему, давая ложную информацию о строке в отладчике!
Есть много вариантов, но общая и также философия по умолчанию, введенная 754, состоит в том, чтобы не заманивать в ловушку, а вместо этого производить специальные результаты, такие как бесконечности, которые могут или не могут проявиться в важных результатах.
В результате функции, которые проверяют состояние отдельных операций, используются не так часто, как функции, которые проверяют представление результатов.
Смотрите, например...
LIST OF FUNCTIONS
Each of the functions that use floating-point values are provided in sin-
gle, double, and extended precision; the double precision prototypes are
listed here. The man pages for the individual functions provide more
details on their use, special cases, and prototypes for their single and
extended precision versions.
int fpclassify(double)
int isfinite(double)
int isinf(double)
int isnan(double)
int isnormal(double)
int signbit(double)
Обновление: для тех, кто действительно думает, что в настоящее время FPU генерирует SIGFPE в случае по умолчанию, я бы посоветовал вам попробовать эту программу. Вы можете легко генерировать переполнение, переполнение и деление на ноль. То, что вы не сгенерируете (если вы не запускаете его на последнем выживающем VAX или не-754 RISC), это SIGFPE:
#include <stdio.h>
#include <stdlib.h>
int main(int ac, char **av) { return printf("%f\n", atof(av[1]) / atof(av[2])); }
В Windows с Visual C++ вы можете контролировать, какие исключения с плавающей запятой не маскируются, используя _control87()
и т. д. Немаскированные исключения с плавающей точкой генерируют структурированные исключения, которые могут быть обработаны с помощью __try
/__except
(и пара других механизмов). Это все полностью зависит от платформы.
Если вы оставите исключения с плавающей запятой замаскированными, другой зависящий от платформы подход к обнаружению этих условий - очистить состояние с плавающей запятой, используя _clear87()
и т.д., выполнить вычисления, а затем запросить состояние с плавающей запятой, используя _status87()
и т. д.
Что-нибудь из этого лучше, чем предложение DigitalRoss проверить результат? В большинстве случаев это не так. Если вам нужно обнаружить (или контролировать) округление (что маловероятно), то, возможно,?
В Windows с Borland/CodeGear/Embarcadero C++ некоторые исключения с плавающей точкой по умолчанию не маскируются, что часто приводит к проблемам при использовании сторонних библиотек, которые не были протестированы с безмаскированными исключениями с плавающей точкой.
Различные компиляторы обрабатывают эти ошибки по-разному.
Неточность почти всегда является результатом деления чисел с абсолютной величиной больше единицы (возможно, через трансцендентные функции). Сложение, вычитание и умножение чисел с абсолютным значением> 1,0 может привести только к переполнению.
Недостаток происходит не очень часто и, вероятно, не будет проблемой в обычных вычислениях, за исключением повторяющихся функций, таких как ряды Тейлора.
Переполнение - это проблема, которая обычно может быть обнаружена с помощью некоего "бесконечного" сравнения, разные компиляторы различаются.
Деление на ноль весьма заметно, так как ваша программа будет (должна) аварийно завершать работу, если у вас нет обработчика ошибок. Проверка дивидендов и делителей поможет избежать проблемы.
Неверные ответы обычно перехватываются без специальных обработчиков ошибок с напечатанной ошибкой DOMAIN.
[РЕДАКТИРОВАТЬ]
Это может помочь: (Руководство по численным вычислениям от Sun) http://docs.sun.com/source/806-3568/
C99 представил функции для обработки исключений с плавающей запятой. До операции с плавающей запятой вы можете использовать feclearexcept()
очистить любые выдающиеся исключения. После операции вы можете использовать fetestexcept()
чтобы проверить, какие флаги исключений установлены.
В Linux вы можете перехватить эти исключения, перехватывая сигнал SIGFPE. Если вы ничего не сделаете, эти исключения прекратят работу вашей программы. Чтобы установить обработчик, используйте функцию сигнала, передавая сигнал, который вы хотите поймать в ловушку, и функцию, которая вызывается в случае срабатывания сигнала.