68960

Генерація виняткових ситуації

Лекция

Информатика, кибернетика и программирование

Якщо виникає необхідність повторно порушити виняткову ситуацію усередині її обробника, можна виконати оператора throw, не указуючи тип виняткової ситуації. В цьому випадку операторові try/catch передається поточна виняткова ситуація. Таким чином для однієї і тієї ж виняткової ситуації...

Украинкский

2014-09-28

56 KB

0 чел.

Лекція № 22

Тема: Генерація виняткових ситуації

План

  1.  Повторне генерування виняткової ситуації
  2.  Функції terminate() і unexpected()
  3.  Обробники, пов'язані з функціями terminate() і unexpected()
  4.  Класи exception і bad_exception

Повторне генерування виняткової ситуації

Якщо виникає необхідність повторно порушити виняткову ситуацію усередині її обробника, можна виконати оператора throw, не указуючи тип виняткової ситуації. В цьому випадку операторові try/catch передається поточна виняткова ситуація. Таким чином для однієї і тієї ж виняткової ситуації можна передбачити декілька обробників. Допустимо, один обробник виняткової ситуації управляє одним аспектом, а другий — останніми. Усередині блоку catch (або функції, що викликається з цього блоку) можна лише повторно згенерувати виняткову ситуацію. В цьому випадку вона передається не тому ж самому операторові catch, а наступному. Повторне генерування виняткової ситуації ілюструється наступною програмою.

 

// Приклад повторного генерування виняткової ситуації.

#include <iostream>

using namespace std;

void Xhandler()

{

try {

throw "Привіт"; // Генерується об'єкт типу char *

} catch(const char *){ // Перехоплення об'єкту типу char *

cout « "Перехоплення об'єкту типу char * усередині функції

Xhandler\n";

throw ; // Повторне генерування об'єкту типу char *

// поза функцією.

}

}

int main() {

cout « "Початок\ п";

try {

Xhandler() ;

} catch(const char *) {

cout << "Перехоплення об'єкту типу char * у функції main\n";

}

cout « "Кінець";

return 0;

}

Результати роботи цієї програми наводяться нижче.

Початок

Перехоплення об'єкту типу char * усередині функції Xhandler()

Перехоплення об'єкту типу char * усередині функції main()

Кінець

Функції terminate() і unexpected()

Як указувалося раніше, функції terminate () і unexpected() викликаються в крайніх випадках, тобто коли обробка виняткової ситуації виконується невірно. Ці функції належать стандартній бібліотеці мови C++. Їх прототипи показані нижче.

void terminate ()

void unexpected()

Для виклику цих функцій необхідний заголовок <exception>.

Функція terminate () викликається, якщо підсистема обробки виняткових ситуацій не може виявити відповідного оператора catch. Крім того, вона викликається, якщо програма намагається повторно згенерувати виняткову ситуацію, яка раніше ніколи не генерувалася. Функція terminate() викликається також в багато інших, заплутаніших ситуаціях. Наприклад, деструкція знищуваного об'єкту генерує виняткову ситуацію в процесі розкручування стека, який виконується при генеруванні іншої виняткової ситуації. Як правило, функція terminate() є останнім засобом обробки виняткової ситуації, якщо ніякою інший обробник не підходить. За умовчанням функція terminate() викликає функцію abort().

Функція unexpected() викликається при спробі генерувати виняткову ситуацію, не вказану в розділі throw. За умовчанням функція unexpected() викликає функцію terminate().

Обробники, пов'язані з функціями terminate() і unexpected()

Функції terminate() і unexpected() просто викликають інші функції, які насправді обробляють помилку. Як показано вище, за умовчанням функція terminate() викликає функцію abort(), а функція unexpected() — функцію terminate(). Таким чином, за умовчанням при виникненні виняткової ситуації обидві функції переривають виконання програми. Проте функції terminate() і unexpected() можуть викликати інші функції. Це дозволяє програмі повністю контролювати підсистему обробки виняткових ситуацій.

Щоб змінити обробник, пов'язаний з функцією terminate (), слід викликати функцію set_terminate (), як показано нижче.

terminate_handler set_terminate (terminate_handler newhandler) throwО;

Тут параметр newhandler є покажчиком на новий обробник, пов'язаний з функцією terminate (). Ця функція повертає покажчик на старий обробник, пов'язаний з функцією terminate(). Новий обробник повинен мати тип terminate_handler, визначений за допомогою наступного оператора:

typedef void (*terminate_handler) ();

Єдине призначення обробника terminate_handler — припинити виконання програми. Він не повинен повертати управління або відновлювати виконання програми.

Щоб змінити обробник, пов'язаний з функцією unexpected (), слід викликати функцію set_unexpected(), як показано нижче.

unexpected_handler set_unexpected(unexpected_handler newhandler) throw();

Тут параметр newhandler є покажчиком на новий обробник, пов'язаний з функцією unexpected(). Ця функція повертає покажчик на старий обробник, пов'язаний з функцією unexpected(). Новий обробник повинен мати тип unexpected_handler, визначений за допомогою наступного оператора:

typedef void (*unexpected_handler)();

Цей обробник може сам генерувати виняткові ситуації, припиняти виконання програми або викликати функцію terminate(). Проте він не повинен повертати управління програмі.

Для виклику функцій set_terminate() і set_unexpected() необхідний заголовок  <expection>.

Розглянемо приклад, в якому визначається обробник, пов'язаний з функцією terminate().

// Визначення нового обробника,

//связанного з функцією terminate.

#include <iostream>

#include <cstdlib>

#include <exception>

using namespace std;

void my_Thandler() {

cout << "Усередині нового обрабочика\ п";

abort();

}

int main() {

// Визначення нового обробника,

// пов'язаного з функцією terminate. set_terminate(my_Thandler);

try {

cout « "Усередині блоку try\n";

throw 100; // Генерування помилки.

} catch (double i) {

// He перехоплює виняткові ситуації типу int.

// ...

}

return 0;

}

Результати цієї програми приведені нижче.

Усередині блоку try

Усередині нового обробника

abnormal program termination

 

Функція uncaught_exception()

Підсистема обробки виняткових ситуацій в мові C++ містить ще одну функцію — uncaught_exception(). Її прототип показаний нижче.

bool uncaught_exception();

Ця функція повертає значення true, якщо виняткова ситуація згенерувала, але ще не перехоплена. Якщо виняткова ситуація була перехоплена, функція повертає значення false.

Класи exception і bad_exception

Функції із стандартної бібліотеки мови C++ можуть генерувати виняткові ситуації, похідні від класу exception. Крім того, обробник, пов'язаний з функцією unexpectedO, може генерувати виняткову ситуацію bad_exception. Для використання цих класів потрібний заголовок <exception>.

Застосування обробки виняткових ситуацій

Система обробки виняткових ситуацій дозволяє реагувати на незвичайні події, що виникають в ході виконання програми. Отже, обробники виняткових ситуацій повинні виконувати якісь розумні дії, які дозволяють виправити помилку або пом'якшити її наслідки. Розглянемо як приклад наступну просту програму. Вона вводить два числа і ділить перше з них на друге. Для запобігання діленню на нуль застосовується обробка виняткової ситуації.

#include <iostream>

using namespace std;

void divide(double а, double b);

int main() {

double i, j;

do {

cout << "Введіть чисельник (6 означає вихід): ";

cin >> i;

cout << "Введіть знаменник: ";

cin >> j;

divide(i, j);

} while(i != 0);

return 0;

}

void divide(double а, double b){

try {

if(!b) throw b; // Перевірка ділення на нуль

cout « "Результат: " << a/b « endl;

} catch (double b) {

cout « "Ділити на нуль не можна.\ n";

}

Цей дуже простій приклад ілюструє принцип обробки виняткових ситуацій. Якщо знаменник рівний нулю, виникає виняткова ситуація. Її обробник не передбачає ділення (це привело б до аварійного завершення роботи програми), а просто повідомляє користувача про виниклу помилку. Таким чином, ділення на нуль можна уникнути, продовживши виконання програми. Ця схема працює і в складніших випадках.

Обробка виняткової ситуації особливо корисна при виході з глибоко вкладених процедур, в яких виникла катастрофічна помилка. Мова C++ надає досконаліші засоби для вирішення цієї проблеми, чим мова C, в якому застосовувалися функції setjmp() і longjmp (). Пам'ятайте, що обробка виняткових ситуацій призначена для забезпечення розумної реакції на виникаючі помилки. Отже, ситуація повинна бути простою і очевидною, наскільки це можливо.


 

А также другие работы, которые могут Вас заинтересовать

18872. Русский конструктивизм. Веснины. Леонидов. Мельников 75.5 KB
  Русский конструктивизм. Веснины. Леонидов. Мельников. Конструктиви́зм советский авангардистский метод стиль направление в изобразительном искусстве архитектуре фотографии и декоративно-прикладном искусстве получивший развитие в 1920 первой половине 1930 годов. Х...
18873. Исторический жанр изобразительного искусства 23.16 KB
  Исторический жанр один из основных жанров изобразительного искусства посвященный воссозданию событий прошлого имеющих историческое значение. Обращённый в основном к прошлому исторический жанр включает также изображение недавних событий историческое значение кото
18874. Ба́ухаус 26.74 KB
  Ба́ухаус нем. Bauhaus Hochschule für Bau und Gestaltung Высшая школа строительства и художественного конструирования или Staatliches Bauhaus учебное заведение существовавшее в Германии с 1919 по 1933 а также художественное объединение возникшее в рамках этого заведения и соответствующее нап...
18875. Художественная сила Нидерландского искусства 33.95 KB
  Пожалуй ни в одной области мировой культуры Нидерланды не сделали столь значительного вклада как в изобразительном искусстве. Произведения нидерландских мастеров XVI XVII вв. до сих пор восхищают зрителей. Голландская художественная школа оказала огромное влияние на все
18876. Архитекту́ра моде́рна 22.21 KB
  Архитекту́ра моде́рна архитектура арнуво архитектурный стиль получивший распространение в Европе в 1890е 1910е годы в рамках художественного направления модерн. Архитектуру модерна отличает отказ от прямых линий и углов в пользу более естественных природных линий...
18877. Символизм в Русском искусстве рубежа XIX-XX вв. М.А. Врубель 290.82 KB
  Символизм в Русском искусстве рубежа XIXXX вв. М.А. Врубель. Врубель М.А. был предшественником символизма. Личность художника объясняет характерную особенность всего отечественного искусства. Это искусство никогда не полагается на холодный расчёт ума. Оно согрето живым ис...
18878. От Модерна к новой архитектуре. Форма следует функции. Франк Ллойд Райт. Органическая архитектура 28.41 KB
  От Модерна к новой архитектуре. Форма следует функции. Франк Ллойд Райт. Органическая архитектура. Декоративные приемы уже в скором времени перестали привлекать архитекторов модерна. Со временем этот стиль изменяется и в нем начинают преобладать недекорированные объ
18879. Импрессионизм. К.Моне, О.Ренуар, Э.Дега 27.76 KB
  Импрессионизм. К.Моне О.Ренуар Э.Дега. История теория основные представители. К. Моне. Завтрак на траве Дама в саду ― поиски новой манеры развитие пленера. 1874г. ― первая выставка импрессионистов происхождение термина. Бульвар капуцинок. Серии: Вокзал Сен Лазар...