std::rethrow_if_nested
Материал из cppreference.com
<tbody>
</tbody>
| Определено в заголовочном файле <exception>
|
||
template< class E > void rethrow_if_nested( const E& e ); |
(начиная с C++11) | |
Если E не является типом полиморфного класса или если std::nested_exception является недоступным или неоднозначным базовым классом E, эффекта нет.
Иначе выполняет
if (auto p = dynamic_cast<const std::nested_exception*>(std::addressof(e)))
p->rethrow_nested();
Параметры
| e | — | объект исключения для повторной генерации |
Возвращаемое значение
(нет)
Примечание
В отличие от многих связанных функций, эта функция не предназначена для вызова с std::exception_ptr, а скорее с фактической ссылкой на исключение.
Возможная реализация
namespace details {
template <class E>
struct can_dynamic_cast
: std::integral_constant<bool,
std::is_polymorphic<E>::value &&
(!std::is_base_of<std::nested_exception, E>::value ||
std::is_convertible<E*, std::nested_exception*>::value)
> { };
template <class T>
void rethrow_if_nested_impl(const T& e, std::true_type)
{
if(auto nep = dynamic_cast<const std::nested_exception*>(std::addressof(e)))
nep->rethrow_nested();
}
template <class T>
void rethrow_if_nested_impl(const T&, std::false_type) { }
}
template <class T>
void rethrow_if_nested(const T& t){
details::rethrow_if_nested_impl(t, details::can_dynamic_cast<T>());
}
|
Пример
Демонстрирует построение и рекурсию вложенного объекта исключения.
Запустить этот код
#include <iostream>
#include <stdexcept>
#include <exception>
#include <string>
#include <fstream>
// выводит поясняющую строку исключения. Если исключение является вложенным,
// рекурсивно выводит пояснение к исключению, которое оно содержит
void print_exception(const std::exception& e, int level = 0)
{
std::cerr << std::string(level, ' ') << "исключение: " << e.what() << '\n';
try {
std::rethrow_if_nested(e);
} catch(const std::exception& nestedException) {
print_exception(nestedException, level+1);
} catch(...) {}
}
// пример функции, которая перехватывает исключение и заключает его во вложенное исключение
void open_file(const std::string& s)
{
try {
std::ifstream file(s);
file.exceptions(std::ios_base::failbit);
} catch(...) {
std::throw_with_nested( std::runtime_error("Не удалось открыть " + s) );
}
}
// пример функции, которая перехватывает исключение и заключает его во вложенное исключение
void run()
{
try {
open_file("nonexistent.file");
} catch(...) {
std::throw_with_nested( std::runtime_error("сбой run()") );
}
}
// запускает пример функции выше и выводит пойманное исключение
int main()
{
try {
run();
} catch(const std::exception& e) {
print_exception(e);
}
}
Возможный вывод:
исключение: сбой run()
исключение: Не удалось открыть nonexistent.file
исключение: basic_ios::clear
Смотрите также
(C++11) |
комбинированный тип для захвата и хранения текущих исключений (класс) |
(C++11) |
бросает свой аргумент с комбинированным std::nested_exception (шаблон функции) |