Важно понимать, что компилятор не проверяет спецификацию noexcept
во время компиляции. Фактически компилятору не разрешено отклонять функцию со спецификатором noexcept
просто потому, что она содержит оператор throw
или вызывает функцию, которая может передавать исключение (однако хорошие компиляторы предупреждают о таких случаях):
//
//
void f() noexcept //
{
throw exception(); //
}
В результате вполне вероятно, что функция, обещавшая не передавать исключений, фактически передаст его. Если такая функция передаст исключение, для соблюдения обещания во время выполнения вызывается функция terminate()
. Результат прокрутки стека непредсказуем. Таким образом, спецификатор noexcept
следует использовать в двух случаях: если есть уверенность, что функция не будет передавать исключений, или если совершенно неизвестно, как справиться с ошибкой.
Спецификация запрета передачи исключения фактически обещает
У прежних версий языка С++ была более сложная схема спецификаций исключения, позволяющая определять типы исключений, которые могла бы передавать функция. Функция может определить ключевое слово throw
, сопровождаемое заключенным в скобки списком типов, которые могла бы передать функция. Спецификатор throw
располагается в том же месте, где и спецификатор noexcept
в текущем языке.
Этот подход никогда широко не использовался и не рекомендован в текущем стандарте. Хотя один случай использования более сложной старой схемы распространен довольно широко. Функция, обозначенная как throw()
, обещает не передавать никаких исключений:
void recoup(int) noexcept; //
void recoup(int) throw(); //
Эти объявления функции recoup()
эквивалентны. Оба указывают, что функция recoup()
не будет передавать исключений.
noexcept
Спецификатор noexcept
получает необязательный аргумент, тип которого должен быть преобразуем в тип bool
: если аргументом будет true
, то функция не будет передавать исключений; если false
— то может:
void recoup(int) noexcept(true); //
void alloc(int) noexcept(false); //
noexcept
noexcept
зачастую создаются с использованием noexcept
. Оператор noexcept
— унарный, возвращающий константное логическое выражение r-значения, означающее способность данного выражения передавать исключения. Подобно оператору sizeof
(см. раздел 4.9), оператор noexcept
не вычисляет свой операнд.
Например, следующее выражение возвращает значение true
:
noexcept(recoup(i)) //
//
поскольку функция recoup()
объявлена со спецификатором noexcept
. В более общем виде выражение noexcept(е)
возвращает значение true
, если у всех вызванных е
функций нет спецификаций передачи и сама е
не содержит операторов throw
. В противном случае выражение noexcept(е)
возвращает значение false
.
Оператор noexcept
можно использовать для формирования спецификатора исключения следующим образом: